详解python tcp编程


Posted in Python onAugust 24, 2020

网络连接与通信是我们学习任何编程语言都绕不过的知识点。 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 连接就建立起来了,后面就是百度向我们传输搜索结果了。

我们来看一个流程图:

详解python tcp编程

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

  • 创建 socket(套接字)
  • 绑定 socket 的 IP 地址和端口号
  • 监听客户端的连接请求
  • 接受客户端的连接请求
  • 与客户端对话
  • 关闭连接

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

  • 创建 socket(套接字)
  • 连接服务器 socket
  • 与服务器对话
  • 关闭连接

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

TCP 客户端

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

# 导入socket库
import socket

# 创建一个socket
s = 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:
    break
data = 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 socket
import threading
import 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)


# 创建 socket
s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
# 监听端口
s.bind(("127.0.0.1", 6000))
# 设定等待连接的最大数量为5
s.listen(5)
print("Waiting for connection...")
# 等待接收连接
while True:
  # 接受一个新连接
  conn, addr = s.accept()
  # 创建新线程来处理TCP连接
  t = threading.Thread(target=tcplink, args=(conn, addr))
  t.start()

客户端代码:

import socket
import time

# 创建 socket
s = 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:53503
Connection 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 通信的过程,熟悉了这个过程是处理后续复杂通信需求的基础。

文中实例代码

以上就是详解python tcp编程的详细内容,更多关于python tcp编程的资料请关注三水点靠木其它相关文章!

Python 相关文章推荐
详解Python中的__getitem__方法与slice对象的切片操作
Jun 27 Python
Python 专题六 局部变量、全局变量global、导入模块变量
Mar 20 Python
python 执行shell命令并将结果保存的实例
May 11 Python
Python实现判断一行代码是否为注释的方法
May 23 Python
Scrapy使用的基本流程与实例讲解
Oct 21 Python
Python线程之定位与销毁的实现
Feb 17 Python
Python批量生成幻影坦克图片实例代码
Jun 04 Python
Python程序包的构建和发布过程示例详解
Jun 09 Python
python实现电子书翻页小程序
Jul 23 Python
Python用input输入列表的实例代码
Feb 07 Python
python爬取网易云音乐热歌榜实例代码
Aug 07 Python
python 删除系统中的文件(按时间,大小,扩展名)
Nov 19 Python
Python rabbitMQ如何实现生产消费者模式
Aug 24 #Python
利用Python的folium包绘制城市道路图的实现示例
Aug 24 #Python
深入分析python 排序
Aug 24 #Python
超级实用的8个Python列表技巧
Aug 24 #Python
基于CentOS搭建Python Django环境过程解析
Aug 24 #Python
详解Django中views数据查询使用locals()函数进行优化
Aug 24 #Python
python3让print输出不换行的方法
Aug 24 #Python
You might like
服务器上配置PHP运行环境教程
2015/02/12 PHP
PHP7.0安装笔记整理
2015/08/28 PHP
PDO的安全处理与事物处理方法
2016/10/31 PHP
jQuery获取文本节点之 text()/val()/html() 方法区别
2011/03/01 Javascript
JS中setInterval、setTimeout不能传递带参数的函数的解决方案
2013/04/28 Javascript
js判断上传文件的类型和大小示例代码
2013/10/18 Javascript
js打开新窗口方法整理
2014/02/17 Javascript
JQuery radio(单选按钮)操作方法汇总
2015/04/15 Javascript
js实现网站最上边可关闭的浮动广告条代码
2015/09/04 Javascript
JavaScript实现的浮动层框架用法实例分析
2015/10/10 Javascript
jQuery的 $.ajax防止重复提交的两种方法(推荐)
2016/10/14 Javascript
headjs实现网站并行加载但顺序执行JS
2016/11/29 Javascript
layer弹窗插件操作方法详解
2017/05/19 Javascript
Easyui在treegrid添加控件的实现方法
2017/06/23 Javascript
JavaScript实现无刷新上传预览图片功能
2017/08/02 Javascript
js保留两位小数方法总结
2018/01/31 Javascript
vue2.0组件之间传值、通信的多种方式(干货)
2018/02/10 Javascript
axios使用拦截器统一处理所有的http请求的方法
2018/11/02 Javascript
Node如何后台数据库使用增删改查功能
2019/11/21 Javascript
详解Vue 数据更新了但页面没有更新的 7 种情况汇总及延伸总结
2020/05/28 Javascript
NodeJS开发人员常见五个错误理解
2020/10/14 NodeJs
js实现鼠标切换图片(无定时器)
2021/01/27 Javascript
Python 3.7新功能之dataclass装饰器详解
2018/04/21 Python
Python进程,多进程,获取进程id,给子进程传递参数操作示例
2019/10/11 Python
如何通过Python3和ssl实现加密通信功能
2020/05/09 Python
Python实现常见的几种加密算法(MD5,SHA-1,HMAC,DES/AES,RSA和ECC)
2020/05/09 Python
python下对hsv颜色空间进行量化操作
2020/06/04 Python
python装饰器代码深入讲解
2021/03/01 Python
德国最大的服装、鞋子和配件在线商店之一:Outfits24
2019/07/23 全球购物
eHarmony英国:全球领先的认真恋爱约会平台之一
2020/11/16 全球购物
婚庆司仪主持词
2014/03/15 职场文书
安全生产专项整治方案
2014/05/06 职场文书
物业总经理助理岗位职责
2014/06/29 职场文书
集结号观后感
2015/06/08 职场文书
2015年文秘个人工作总结
2015/10/14 职场文书
HTML页面点击按钮关闭页面的多种方式
2022/12/24 HTML / CSS