Python各协议下socket黏包问题原理


Posted in Python onApril 12, 2022

1.socket黏包问题原理

黏包:指数据与数据之间没有明确的分界线,导致不能正确的读取数据。

应用数据想要发送数据就必须将数据交给操作系统,而操作系统需要同时为所有的应用程序提供数据传输服务,就意味着不可能马上将应用数据发送,就需要为程序提供一个缓冲区,用于临时存放数据。

当发送数据很快,有两条数据都在缓冲区时,操作系统可能将两个数据发给接收方,数据之间没有分界线,接收方会误认为是一条数据。

2.UDP协议

UDP在收发数据时是基于数据包的,即一个包一个包的发送,包与包之间有明确的分界,到达对方缓冲区后也是独立数据包。这种方式存在的问题:

①发送数据的长度每个操作系统会有不同的限制,数据超过限制则无法发送;

②接收方接收数据时,如果应用程序提供的缓存容量小于数据包的长度,则会造成数据的丢失,而缓冲区大小不可能无限大。

这意味着UDP不会出现黏包问题,但会丢失数据,不可靠。

3.TCP协议

TCP增加了一套校验规则来保证数据的完整性,会将超过TCP包最大长度的数据拆分为多个TCP包,并在传输数据时为每一个TCP数据包指定一个顺序号,接收方在收到TCP数据包后按照顺序将数据包进行重组,重组后的数据全都是二进制数据,且每次收到的二进制数据之间没有明显的分界。基于这种工作机制,TCP在三种情况下发生黏包问题:

①当单个数据包较小时,接收方可能一次性读取了多个包的数据;

②当整体数据较大时,接收方可能一次性仅读取了一个包的一部分内容;

③另外TCP协议为提高效率,增加了一种优化机制,会将数据小且发送间隔短的数据合并发送,该机制也会导致发送方将两个数据包粘在一起发送。

也就是说,TCP传输数据是可靠的,但是会黏包。

4.发送方出现的黏包

服务器端:

from socket import *
server_socket = socket(AF_INET,SOCK_STREAM)
server_socket.bind(('',8080))
server_socket.listen(5)
 
new_socket,client_addr = server_socket.accept()
 
data1 = new_socket.recv(1024)
data2 = new_socket.recv(1024)
print("收到的第一条数据:",data1)
print("收到的第二条数据:",data2)
 
new_socket.close()
server_socket.close()

客户端:

from socket import *
 
client_socket = socket(AF_INET,SOCK_STREAM)
client_socket.connect(('10.175.193.126',8080))
client_socket.send('hello'.encode('utf-8'))
client_socket.send('word'.encode('utf-8'))
client_socket.close()

服务器端接收到的数据:

Python各协议下socket黏包问题原理

由于客户端两条数据发送间隔太短且数据包太小,被服务器端误认为是一条数据。

5. 接收方出现的黏包

服务器端:

from socket import *
import time
 
server_socket = socket(AF_INET,SOCK_STREAM)
server_socket.bind(('',8080))
server_socket.listen(5)
 
new_socket,client_addr = server_socket.accept()
print("连接成功!",client_addr)
 
data1 = new_socket.recv(3) #每次只接收三个字节,接收不完整
time.sleep(6)
print("收到的第一条数据:",data1)
 
data2 = new_socket.recv(10)
#接收第一次未接收的数据,若有空间,会继续接收新数据
print("收到的第二条数据:",data2)
 
new_socket.close()
server_socket.close()

客户端:

from socket import *
#通过time模块使客户端发送多个数据包时,时间间隔变长
import time
 
client_socket = socket(AF_INET,SOCK_STREAM)
client_socket.connect(('10.175.193.126',8080))
client_socket.send('hello'.encode('utf-8'))
time.sleep(5) #让当前线程休眠5秒
client_socket.send('word'.encode('utf-8'))
 
client_socket.close()

服务器端接收到的数据:

Python各协议下socket黏包问题原理

6.黏包的成因

①服务器端出现黏包:接收方不知道消息之间的界限,不知道一个消息要提取多少字节的数据造成的;

②客户端出现黏包:TCP在发送数据少且间隔时间短的数据包时,会将几条合并一起发送。

到此这篇关于Python数据传输黏包问题的文章就介绍到这了!

Python 相关文章推荐
在Python的Bottle框架中使用微信API的示例
Apr 23 Python
Python使用SocketServer模块编写基本服务器程序的教程
Jul 12 Python
python 3.6 +pyMysql 操作mysql数据库(实例讲解)
Dec 20 Python
python中实现将多个print输出合成一个数组
Apr 19 Python
Python中三元表达式的几种写法介绍
Mar 04 Python
python机器学习库scikit-learn:SVR的基本应用
Jun 26 Python
python 获取sqlite3数据库的表名和表字段名的实例
Jul 17 Python
Django 解决开发自定义抛出异常的问题
May 21 Python
如何用python处理excel表格
Jun 09 Python
Python列表嵌套常见坑点及解决方案
Sep 30 Python
python 三种方法提取pdf中的图片
Feb 07 Python
python中random模块详解
Mar 01 Python
Python爬虫网络请求之代理服务器和动态Cookies
Apr 12 #Python
分享Python异步爬取知乎热榜
尝试使用Python爬取城市租房信息
Apr 12 #Python
Python采集爬取京东商品信息和评论并存入MySQL
Apr 12 #Python
Python实现批量将文件复制到新的目录中再修改名称
Python多线程实用方法以及共享变量资源竞争问题
Apr 12 #Python
Python使用socket去实现TCP客户端和TCP服务端
Apr 12 #Python
You might like
聊天室php&mysql(二)
2006/10/09 PHP
Yii使用find findAll查找出指定字段的实现方法
2014/09/05 PHP
php rsa加密解密使用详解
2015/01/14 PHP
php实现用于删除整个目录的递归函数
2015/03/16 PHP
php创建无限级树型菜单
2015/11/05 PHP
PHP实现提高SESSION响应速度的几种方法详解
2019/08/09 PHP
innerText和innerHTML 一些问题分析
2009/05/18 Javascript
js substr支持中文截取函数代码(中文是双字节)
2013/04/17 Javascript
javascript中数组的多种定义方法和常用函数简介
2014/05/09 Javascript
js加密解密字符串可自定义密码因子
2014/05/13 Javascript
bootstrap输入框组使用方法
2017/02/07 Javascript
AngularJS的ng-click传参的方法
2017/06/19 Javascript
Node.js+jade抓取博客所有文章生成静态html文件的实例
2017/09/19 Javascript
移动前端图片压缩上传的实例
2017/12/06 Javascript
JS实现的JSON数组去重算法示例
2018/04/11 Javascript
vue项目首屏打开速度慢的解决方法
2019/03/31 Javascript
vue 项目打包时样式及背景图片路径找不到的解决方式
2019/11/12 Javascript
vue实现路由不变的情况下,刷新页面操作示例
2020/02/02 Javascript
python实现马耳可夫链算法实例分析
2015/05/20 Python
浅谈python中的数字类型与处理工具
2017/08/02 Python
用tensorflow搭建CNN的方法
2018/03/05 Python
Python中的pack和unpack的使用
2018/03/12 Python
python实现决策树、随机森林的简单原理
2018/03/26 Python
Python3之读取连接过的网络并定位的方法
2018/04/22 Python
Pyqt5实现英文学习词典
2019/06/24 Python
使用Python opencv实现视频与图片的相互转换
2019/07/08 Python
Python几种常见算法汇总
2020/06/02 Python
Python实现迪杰斯特拉算法过程解析
2020/09/18 Python
阿里云:Aliyun.com
2017/02/15 全球购物
女孩每月服装订阅盒:kidpik
2019/04/17 全球购物
英国办公家具网站:Furniture At Work
2019/10/07 全球购物
机电专业大学生求职信
2013/10/04 职场文书
护士自我鉴定
2013/10/23 职场文书
上班迟到检讨书
2014/09/15 职场文书
个人自我剖析材料
2014/09/30 职场文书
7个你应该知道的JS原生错误类型
2021/04/29 Javascript