Python之网路编程之粘包现象
一、什么是粘包
须知:只有TCP有粘包现象,UDP永远不会粘包
粘包不一定会发生
如果发生了:1.可能是在客户端已经粘了
2.客户端没有粘,可能是在服务端粘了
首先需要掌握一个socket收发消息的原理

应用程序所看到的数据是一个整体,或说是一个流(stream),一条消息有多少字节对应用程序是不可见的,因此TCP协议是面向流的协议,这也是容易出现粘包问题的原因。(因为TCP是流式协议,不知道啥时候开始,啥时候结束)。而UDP是面向消息的协议,每个UDP段都是一条消息,应用程序必须以消息为单位提取数据,不能一次提取任意字节的数据,这一点和TCP是很不同的。怎样定义消息呢?可以认为对方一次性write/send的数据为一个消息,需要明白的是当对方send一条信息的时候,无论底层怎样分段分片,TCP协议层会把构成整条消息的数据段排序完成后才呈现在内核缓冲区。
所谓粘包问题主要还是因为接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所造成的。
二、发生粘包的两种情况
发送端需要等缓冲区满才发送出去,造成粘包(发送数据时间间隔很短,数据了很小,会当做一个包发出去,产生粘包)
1 from socket import *
2 phone = socket(AF_INET,SOCK_STREAM)
3 phone.setsockopt(SOL_SOCKET,SOCK_STREAM,1)
4 phone.bind(('127.0.0.1',8080))
5 phone.listen(5)
6 print('start running...')
7
8 coon,addr = phone.accept() #等待连接
9
10 data1 = coon.recv(10)
11 data2 = coon.recv(10)
12
13 print('------------>',data1.decode('utf-8'))
14 print('------------>',data2.decode('utf-8'))
15 coon.close()
16 phone.close()
服务端
1 from socket import *
2 import time
3 phone = socket(AF_INET,SOCK_STREAM)
4 phone.connect(('127.0.0.1',8080))
5
6 phone.send('hello'.encode('utf-8'))
7 phone.send('helloworld'.encode('utf-8'))
8 phone.close()
客户端
接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包)
1 from socket import *
2 phone = socket(AF_INET,SOCK_STREAM)
3 phone.setsockopt(SOL_SOCKET,SOCK_STREAM,1)
4 phone.bind(('127.0.0.1',8080))
5 phone.listen(5)
6 print('start running...')
7
8 coon,addr = phone.accept() #等待连接
9
10 data1 = coon.recv(2) #一次没有接收完整
11 data2 = coon.recv(10) #下一次接收的时候会先取旧的数据,然后取新的
12 # data3 = coon.recv(1024) #接收等5秒后的信息
13 print('------------>',data1.decode('utf-8'))
14 print('------------>',data2.decode('utf-8'))
15 # print('------------>',data3.decode('utf-8'))
16 coon.close()
17 phone.close()
服务端
1 from socket import *
2 import time
3 phone = socket(AF_INET,SOCK_STREAM)
4 phone.connect(('127.0.0.1',8080))
5
6 phone.send('hello'.encode('utf-8'))
7 time.sleep(5)
8 phone.send('haiyan'.encode('utf-8'))
9 phone.close()
客户端
三、解决粘包的方法
问题的根源在于,接收端不知道发送端将要传送的字节流的长度,所以解决粘包的方法就是围绕,如何让发送端在发送数据前,把自己将要发送的字节流总大小让接收端知晓,然后接收端来一个死循环接收完所有数据
1 import socket
2 import subprocess
3 import struct
4 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM) #买手机
5 phone.bind(('127.0.0.1',8080)) #绑定手机卡
6 phone.listen(5) #阻塞的最大数
7 print('start runing.....')
8 while True: #链接循环
9 coon,addr = phone.accept()# 等待接电话
10 print(coon,addr)
11 while True: #通信循环
12 # 收发消息
13 cmd = coon.recv(1024) #接收的最大数
14 print('接收的是:%s'%cmd.decode('utf-8'))
15 #处理过程
16 res = subprocess.Popen(cmd.decode('utf-8'),shell = True,
17 stdout=subprocess.PIPE, #标准输出
18 stderr=subprocess.PIPE #标准错误
19 )
20 stdout = res.stdout.read()
21 stderr = res.stderr.read()
22 #先发报头(转成固定长度的bytes类型,那么怎么转呢?就用到了struct模块)
23 #len(stdout) + len(stderr)#统计数据的长度
24 header = struct.pack('i',len(stdout)+len(stderr))#制作报头
25 coon.send(header)
26 #再发命令的结果
27 coon.send(stdout)
28 coon.send(stderr)
29 coon.close()
30 phone.close()
服务端
1 import socket
2 import struct
3 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
4 phone.connect(('127.0.0.1',8080)) #连接服
5 while True:
6 # 发收消息
7 cmd = input('请你输入命令>>:').strip()
8 if not cmd:continue
9 phone.send(cmd.encode('utf-8')) #发送
10 #先收报头
11 header_struct = phone.recv(4) #收四个
12 unpack_res = struct.unpack('i',header_struct)
13 total_size = unpack_res[0] #总长度
14 #后收数据
15 recv_size = 0
16 total_data=b''
17 while recv_size<total_size: #循环的收
18 recv_data = phone.recv(1024) #1024只是一个最大的限制
19 recv_size+=len(recv_data) #
20 total_data+=recv_data #
21 print('返回的消息:%s'%total_data.decode('gbk'))
22 phone.close()
客户端
四、解决粘包问题升级版:完整的解决了
1 import socket
2 import subprocess
3 import struct
4 import json
5 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM) #买手机
6 phone.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
7 phone.bind(('127.0.0.1',8080)) #绑定手机卡
8 phone.listen(5) #阻塞的最大数
9 print('start runing.....')
10 while True: #链接循环
11 coon,addr = phone.accept()# 等待接电话
12 print(coon,addr)
13 while True: #通信循环
14 # 收发消息
15 cmd = coon.recv(1024) #接收的最大数
16 print('接收的是:%s'%cmd.decode('utf-8'))
17 #处理过程
18 res = subprocess.Popen(cmd.decode('utf-8'),shell = True,
19 stdout=subprocess.PIPE, #标准输出
20 stderr=subprocess.PIPE #标准错误
21 )
22 stdout = res.stdout.read()
23 stderr = res.stderr.read()
24 # 制作报头
25 header_dic = {
26 'total_size': len(stdout)+len(stderr), # 总共的大小
27 'filename': None,
28 'md5': None
29 }
30 header_json = json.dumps(header_dic) #字符串类型
31 header_bytes = header_json.encode('utf-8') #转成bytes类型(但是长度是可变的)
32 #先发报头的长度
33 coon.send(struct.pack('i',len(header_bytes))) #发送固定长度的报头
34 #再发报头
35 coon.send(header_bytes)
36 #最后发命令的结果
37 coon.send(stdout)
38 coon.send(stderr)
39 coon.close()
40 phone.close()
服务端
1 import socket
2 import struct
3 import json
4 phone = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
5 phone.connect(('127.0.0.1',8080)) #连接服务器
6 while True:
7 # 发收消息
8 cmd = input('请你输入命令>>:').strip()
9 if not cmd:continue
10 phone.send(cmd.encode('utf-8')) #发送
11 #先收报头的长度
12 header_len = struct.unpack('i',phone.recv(4))[0] #吧bytes类型的反解
13 #在收报头
14 header_bytes = phone.recv(header_len) #收过来的也是bytes类型
15 header_json = header_bytes.decode('utf-8') #拿到json格式的字典
16 header_dic = json.loads(header_json) #反序列化拿到字典了
17 total_size = header_dic['total_size'] #就拿到数据的总长度了
18 #最后收数据
19 recv_size = 0
20 total_data=b''
21 while recv_size<total_size: #循环的收
22 recv_data = phone.recv(1024) #1024只是一个最大的限制
23 recv_size+=len(recv_data) #有可能接收的不是1024个字节,或许比1024多呢,
24 # 那么接收的时候就接收不全,所以还要加上接收的那个长度
25 total_data+=recv_data #最终的结果
26 print('返回的消息:%s'%total_data.decode('gbk'))
27 phone.close()
客户端
五、struct模块
1 #该模块可以把一个类型,如数字,转成固定长度的bytes类型
2 import struct
3 res = struct.pack('i',12345)
4 print(res,len(res),type(res)) #长度是4
5
6 res2 = struct.pack('i',12345111)
7 print(res,len(res),type(res2)) #长度也是4
8
9 unpack_res =struct.unpack('i',res2)
10 print(unpack_res) #(12345111,)
11 print(unpack_res[0]) #
struct
Python之网路编程之粘包现象的更多相关文章
- python socket网络编程之粘包问题详解
一,粘包问题详情 1,只有TCP有粘包现象,UDP永远不会粘包 你的程序实际上无权直接操作网卡的,你操作网卡都是通过操作系统给用户程序暴露出来的接口,那每次你的程序要给远程发数据时,其实是先把数据从用 ...
- Python网络编程04 /recv工作原理、展示收发问题、粘包现象
Python网络编程04 /recv工作原理.展示收发问题.粘包现象 目录 Python网络编程04 /recv工作原理.展示收发问题.粘包现象 1. recv工作原理 2. 展示收发问题示例 发多次 ...
- Python网络编程(2)-粘包现象及socketserver模块实现TCP并发
1. 基于Tcp的远程调用命令实现 很多人应该都使用过Xshell工具,这是一个远程连接工具,通过上面的知识,就可以模拟出Xshell远程连接服务器并调用命令的功能. Tcp服务端代码如下: impo ...
- Python之路(第三十一篇) 网络编程:简单的tcp套接字通信、粘包现象
一.简单的tcp套接字通信 套接字通信的一般流程 服务端 server = socket() #创建服务器套接字 server.bind() #把地址绑定到套接字,网络地址加端口 server.lis ...
- python网络编程基础之socket粘包现象
粘包现象两种 登陆 #服务端import json import socket server=socket.socket()#创建socket对象 ip_port=('127.0.0.1',8001) ...
- python/socket编程之粘包
python/socket编程之粘包 粘包 只有TCP有粘包现象,UDP永远不会粘包. 首先需要掌握一个socket收发消息的原理 发送端可以是1k,1k的发送数据而接受端的应用程序可以2k,2k的提 ...
- Python之路 - 网络编程之粘包
Python之路 - 网络编程之粘包 粘包
- python socket的应用 以及tcp中的粘包现象
1,socket套接字 一个接口模块,在tcp/udp协议之间的传输接口,将其影藏在socket之后,用户看到的是socket让其看到的. 在tcp中当做server和client的主要模块运用 #s ...
- 网络编程-SOCKET开发之----2. TCP粘包现象产生分析
1. 粘包现象及产生原因 1)概念 指TCP协议中,发送方发送的若干个包数据到接收方接收时粘成一包.发送方粘包:发送方把若干个要发送的数据包封装成一个包,一次性发送,减少网络IO延迟:接收方粘包:接收 ...
随机推荐
- vim技巧1
在编辑模式或可视模式下输入的命令会另外注明.1. 查找 /xxx(?xxx) 表示在整篇文档中搜索匹配xxx的字符串, / 表示向下查找, ? 表示 ...
- delphi application 和 form 主窗体 都是窗口
application 也是一个窗体, 和一般窗体不一样. 主窗体 是一个窗体. 然后这两个窗体,分别计算pid 获得结果竟然一样. 另外句柄是动态,全部都不一样.每一次都不一样.
- mingw32环境上静态编译 dav1d 0.4.0
2019-08-05 发布 dav1d 0.4.0 dav1d 'Cheetah', the fast and small AV1 decoder This is the fourth major r ...
- c++ 引用 日期&时间
日期时间[点击进入看吧,没啥可后期拓展的] 引用 引用变量是一个别名,也就是说,它是某个已存在变量的另一个名字.一旦把引用初始化为某个变量,就可以使用该引用名称或变量名称来指向变量. 一.引用和指针的 ...
- 【HANA系列】SAP HANA的ini文件存储路径
公众号:SAP Technical 本文作者:matinal 原文出处:http://www.cnblogs.com/SAPmatinal/ 原文链接:[HANA系列]SAP HANA的ini文件存储 ...
- Fescar使用(资料)
fescar源码走读1:业务调用方 https://zhuanlan.zhihu.com/p/54659540 fescar源码走读2:fescar服务端 https://zhuanlan.zhi ...
- mysql for update 高并发 死锁研究
mysql for update语句 https://www.cnblogs.com/jtlgb/p/8359266.html For update带来的思考 http://www.cnblo ...
- 你确定 SQL 查询都是以 SELECT 开始的?
很多 SQL 查询都是以 SELECT 开始的. 不过,最近我跟别人解释什么是窗口函数,我在网上搜索"是否可以对窗口函数返回的结果进行过滤"这个问题,得出的结论是"窗口函 ...
- laravel的monolog使用
Laravel 集成了 Monolog 日志函数库,Monolog 支持和提供多种强大的日志处理功能. 1.设置,日志模式 (1)Laravel 提供可立即使用的 single.daily.syslo ...
- [LeetCode] 227. 基本计算器 II
题目链接: https://leetcode-cn.com/problems/basic-calculator-ii 难度:中等 通过率:33.2% 题目描述: 实现一个基本的计算器来计算一个简单的字 ...