Python Tcp编程

网络连接与通信是我们学习任何编程语言都绕不过的知识点。Python 也不例外,本文就介绍因特网的核心协议 TCP ,以及如何用 Python 实现 TCP 的连接与通信。

TCP 协议

TCP协议(Transmission Control Protocol, 传输控制协议)是一种面向连接的传输层通信协议,它能提供高可靠性通信,像 HTTP/HTTPS 等网络服务都采用 TCP 协议通讯。那么网络通讯方面都会涉及到 socket 编程,当然也包括 TCP 协议。

Network Socket

我们来看看定义:

Network Socket(网络套接字)是计算机网络中进程间通信的数据流端点,广义上也代表操作系统提供的一种进程间通信机制。

这些计算机术语都很学术,难于理解,每个字都认识,加在一起就不认识了。我们可以通俗地理解成发快递:A 需要给 B 寄快递,首先需要知道 B 的地址和手机号码,那么这个地址就相当于 网络中的主机 IP 地址,而手机就相当于 主机的端口号。然后 A 还需要指定哪家快递公司,是顺丰还是中通?这个快递公司就相当于通信的传输协议。

TCP 连接流程

上述快递的例子中,寄快递的我们可以叫做客户端,收快递的我们叫做服务器。专业点就是主动发起连接的一方叫做客户端,被动响应的一方叫做服务器。例如,我们在浏览器中访问百度搜索时,我们自己的电脑就是客户端,浏览器会向百度的服务器发送连接请求,如果百度的服务器接受了我们的请求,那么一个 TCP 连接就建立起来了,后面就是百度向我们传输搜索结果了。

我们来看一个流程图:

图片

TCP服务器的建立可以归纳这几步:

  • 创建 socket(套接字)

  • 绑定 socket 的 IP 地址和端口号

  • 监听客户端的连接请求

  • 接受客户端的连接请求

  • 与客户端对话

  • 关闭连接

TCP客户端的创建可总结为这几步:

  • 创建 socket(套接字)

  • 连接服务器 socket

  • 与服务器对话

  • 关闭连接

这里需要注意的是 TCP 客户端连接到服务器的 IP 和端口号必须是 TCP 服务器的 IP 和监听的端口号,服务器调用 listen() 开始监听端口,然后调用 accept() 时刻准备接受客户端的连接请求,此时服务器处于阻塞状态,直到服务器监听到客户端的请求后,接收请求并建立连接为止。

TCP 客户端

创建 socket 连接,可以这样做:

# 导入socket库import socket# 创建一个sockets = socket.socket(socket.AF_INET, socket.SOCK_STREAM)# 建立连接s.connect(("127.0.0.1", 6000))

创建 socket 时,第一个参数 socket.AF_INET 表示指定使用 IPv4 协议,如果要使用 IPv6 协议,就指定为 socket.AF_INET6。SOCK_STREAM 指定使用面向流的 TCP 协议。然后我们调用 connect() 方法,传入 IP 地址(或者域名),指定端口号就可以建立连接了。

接下来我们就可以向服务器发送数据了:

s.send(b'Hello, Mr Right!')

接收数据时,调用 recv(max) 方法,一次最多接收指定的字节数,因此,在一个 while 循环中反复接收,直到 recv() 返回空数据,表示接收完毕,退出循环。​​​​​​​

# 接收数据buffer = []while True:    # 每次最多接收1k字节    d = s.recv(1024)    if d:        buffer.append(d)    else:        breakdata = b''.join(buffer)

最后,我们需要关闭连接,很简单:

s.close()

TCP 服务器

相比于客户端,服务器端稍微复杂一些,需要先绑定一个 IP 地址和端口号,然后监听客户端的请求,收到请求后丢到一个线程去处理。

创建 socket 跟客户端方法一样:

s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)

接下来需要绑定监听地址和端口:

s.bind(('127.0.0.1', 6000))

然后就可以开始监听端口了,监听时需要传入一个参数,指定等待连接的最大数量:

s.listen(5)

接下来就是无限循环等待客户端的连接,直到有连接请求过来,就用一个线程去处理:​​​​​​​

while True:    # 接受一个新连接    sock, addr = s.accept()    # 创建新线程来处理TCP连接    t = threading.Thread(target=tcplink, args=(sock, addr))    t.start()

这里为什么需要多线程处理呢?想象一下菜鸟驿站,如果里面只有一个人的话,那么多个人寄件就需要排队,一个个来;但是如果有多个人的话,那么每个人都可以处理一个寄件请求。

我们来看一下处理客户端请求的方法:​​​​​​​

# 处理tcp连接def tcplink(conn, addr):    print("Accept new connection from %s:%s" % addr)    # 向客户端发送欢迎消息    conn.send(b"Server: Welcome!\n")    while True:        conn.send(b"Server: What's your name?")        data = conn.recv(1024)        # 如果客户端发送 exit 过来请求退出,结束循环        if data == b"exit":            conn.send(b"Server: Good bye!\n")            break        conn.send(b"Server: Hello %s!\n" % data)    # 关闭连接    conn.close()    print("Connection from %s:%s is closed" % addr)

例子中,我们先想客户端发送欢迎消息,然后询问客户端名称,收到名称后发送欢迎消息,直到接收到客户端的 'exit' 命令,退出循环,关闭连接。

实例

我们把上面的分步讲解代码合并起来,形成一个可运行的实例。

服务器端代码:​​​​​​​

import socketimport threadingimport time# 处理tcp连接def tcplink(conn, addr):    print("Accept new connection from %s:%s" % addr)    # 向客户端发送欢迎消息    conn.send(b"Server: Welcome!\n")    while True:        conn.send(b"Server: What's your name?")        data = conn.recv(1024)        # 如果客户端发送 exit 过来请求退出,结束循环        if data == b"exit":            conn.send(b"Server: Good bye!\n")            break        conn.send(b"Server: Hello %s!\n" % data)    time.sleep(5)    # 关闭连接    conn.close()    print("Connection from %s:%s is closed" % addr)# 创建 sockets = socket.socket(socket.AF_INET, socket.SOCK_STREAM)# 监听端口s.bind(("127.0.0.1", 6000))# 设定等待连接的最大数量为5s.listen(5)print("Waiting for connection...")# 等待接收连接while True:    # 接受一个新连接    conn, addr = s.accept()    # 创建新线程来处理TCP连接    t = threading.Thread(target=tcplink, args=(conn, addr))    t.start()

客户端代码:​​​​​​​

import socketimport time# 创建 sockets = socket.socket(socket.AF_INET, socket.SOCK_STREAM)# 建立连接s.connect(("127.0.0.1", 6000))# 接收服务器消息print(s.recv(1024).decode())for data in [b'Michael', b'Tracy', b'Sarah']:    # 发送数据    s.send(data)    time.sleep(2)    # 打印接收到的数据    print(s.recv(1024).decode('utf-8'))    time.sleep(1)time.sleep(3)# 请求退出s.send(b'exit')time.sleep(2)print(s.recv(1024).decode('utf-8'))# 关闭连接s.close()

注意,在代码中,我加入了一些休眠(sleep)操作,主要是为了控制台能够顺利打印出来,不然程序运行太快,打印顺序和内容有可能和预期不一样。

先运行服务器端代码,然后再运行客户端代码,我们可以看到服务器端控制台打印内容如下:# 服务器端打印消息

Waiting for connection...Accept new connection from 127.0.0.1:53503Connection from 127.0.0.1:53503 is closed

客户端控制台打印内容如下:​​​​​​​

# 客户端打印消息Server: Welcome!Server: What's your name?Server: Hello Michael!Server: What's your name?Server: Hello Tracy!Server: What's your name?Server: Hello Sarah!Server: What's your name?Server: Good bye!

大家可以对照着打印内容和代码,体会一下服务器端和客户端通信的原理。

总结

本文为大家介绍了 TCP 编程的基本原理和如何使用 Python 实现一个最简单的 TCP 通信过程。通过介绍和实例,大家要在脑海中形成一个 TCP 通信的过程,熟悉了这个过程是处理后续复杂通信需求的基础。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/95817.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微信小程序左上角home图标的解决方法之一 层级混乱导致的home图标显示的问题 自定义左上角左侧图标的返回路径

这个项目的编辑页在tabbar上 导致跳到tabbar得使用wx.switchTab 保存后返回原来的页面就出现了左上角的home图标 本来想通过自定义home图标的跳转路径来解决这个问题 没想到居然找不到相关内容 有清楚的朋友麻烦给我留个言不胜感激 那我写一下我的骚操作 app.js globalData: {…

Git学习——细节补充

Git学习——细节补充 1. git diff2. git log3. git reset4. git reflog5. 提交撤销5.1 当你改乱了工作区某个文件的内容,想直接丢弃工作区的修改时5.2 当提交到了stage区后,想要退回 6. git remote7. git pull origin master --no-rebase8. 分支管理9. g…

【计算机网络】序列化与反序列化

文章目录 1. 如何处理结构化数据?序列化 与 反序列化 2. 实现网络版计算器1. Tcp 套接字的封装——sock.hpp创建套接字——Socket绑定——Bind将套接字设置为监听状态——Listen获取连接——Accept发起连接——Connect 2. 服务器的实现 ——TcpServer.hpp初始化启动…

uniapp 微信小程序 获取用户头像和昵称

一、背景 自2022年10月25日后,小程序 wx.getUserProfile 接口 被收回,通过 wx.getUserInfo 接口获取用户头像将统一返回默认灰色头像,昵称将统一返回 “微信用户”。如需获取用户头像昵称,可以手动获取,具体步骤&…

微信 小程序 在电脑PC端无法加载的解决办法。电脑微信小程序打不开是怎么回事?电脑微信小程序不能打开解决方法教学

一、电脑微信小程序打不开或者一直在加载的原因? 1、电脑端微信版本未更新 微信版本未及时更新,也会影响小程序的正常打开,可以尝试更新版本。 2、缓存过多 如果电脑缓存文件过多,内存少,也可能导致小程序无法流畅…

Java-Optional类

概述 Optional是JAVA 8引入的一个类,用于处理可能为null的值。 利用Optional可以减少代码中if-else的判断逻辑,增加代码的可读性。且可以减少空指针异常的发生,增加代码的安全性。 常用的方法 示例 代码 public class OptionalTest {pub…

ARM 汇编基础知识

1.为什么学习汇编? 我们在进行嵌入式 Linux 开发的时候是绝对要掌握基本的 ARM 汇编,因为 Cortex-A 芯片一 上电 SP 指针还没初始化, C 环境还没准备好,所以肯定不能运行 C 代码,必须先用汇编语言设置好 C 环境…

【LeetCode算法系列题解】第21~25题

CONTENTS LeetCode 21. 合并两个有序链表(简单)LeetCode 22. 括号生成(中等)LeetCode 23. 合并K个升序链表(困难)LeetCode 24. 两两交换链表中的节点(中等)LeetCode 25. K 个一组翻转…

uni-app:允许字符间能自动换行(英文字符、数字等)

<template><view class"container"><!-- 这里是你的文本内容 -->{{ multilineText }}</view> </template><style> .container {word-break: break-all; } </style>例如&#xff1a; <template><view class"…

Spring Cloud--从零开始搭建微服务基础环境【三】

&#x1f600;前言 本篇博文是关于Spring Cloud–从零开始搭建微服务基础环境【三】&#xff0c;希望你能够喜欢 &#x1f3e0;个人主页&#xff1a;晨犀主页 &#x1f9d1;个人简介&#xff1a;大家好&#xff0c;我是晨犀&#xff0c;希望我的文章可以帮助到大家&#xff0c;…

实战-支付漏洞

免责声明 本文发布的工具和脚本&#xff0c;仅用作测试和学习研究&#xff0c;禁止用于商业用途&#xff0c;不能保证其合法性&#xff0c;准确性&#xff0c;完整性和有效性&#xff0c;请根据情况自行判断。如果任何单位或个人认为该项目的脚本可能涉嫌侵犯其权利&#xff0c…

pdf怎么删除其中一页?

pdf怎么删除其中一页&#xff1f;现在&#xff0c;pdf文件已经深入影响着我们的工作和学习&#xff0c;如果你是一个上班族&#xff0c;那么几乎每天都会使用到pdf格式的电脑文件。当我们阅读一个页数众多的PDF文件时&#xff0c;可能会发现实际上只需要其中的一小部分内容。很…