python 套接字 struck_Python socket粘包问题(最终解决办法)

套接字:

就是将传输层以下的协议封装成子接口

对于应用程序来说只需调用套接字的接口,写出的程序自然是遵循tcp或udp协议的

实现第一个功能个:

实现:通过客户端向服务端发送命令,调取windows下面的cmd窗口,将服务端执行命令的结构,返回并显示在

客户端窗口上。

subprocess:

1.可以将执行结果返回

2.返回值是bytes类型

(基于这两点,可以应用在server端,将服务端的返回直接以bytes的格式直接send给客户端,

实现在客户端的显示)

问题1:粘包问题

粘包问题:实际是由TCP协议在传输数据时的2大特性来的

TCP协议又叫流式协议,在数据传输上,只要客户端发送相应的数据请求,

服务端会将数据像流水一样不断的send给客户端

基于这个特点,就会存在一个问题,当客户端给服务端发送一条命令,服务端成功接收并将命令的

结果返回到客户端的时候,由于客户端recv()的数量限制,可以一次不能完全取出,

这个时候就会存在,下次输入命令客户端首先拿到的返回值就是上次残留的没有收完的数据

基于粘包问题的解决思路就是:

发数据之前先把报头发给对方,让对方先知道要收的报头的长度,后面再传数据文件

自定义报头:

为甚么要自定义报头:

因为struck path(‘i’,3443242)

1.’i‘:类型不同,后面数字的长度大小也不同,大小是有限的(当超出范围时会报错)

2.因为报头里面含有的内容可能不仅仅只有total_siz还有filename、hash等等,知识单纯的把total_size

当做报头传入不合理,所以我们要自定义报头

Server端配置

from socket import *

importsocket,subprocess,struct,json

server=socket.socket(AF_INET, SOCK_STREAM)

server.bind(('127.0.0.1', 8080))

server.listen(5)whileTrue:

conn,client=server.accept()print(client)whileTrue:try:

cmd= conn.recv(1024)if len(cmd) == 0: breakobj=subprocess.Popen(

cmd.decode('utf-8'),

shell=True,

stdout=subprocess.PIPE,

stderr=subprocess.PIPE,

)

out=obj.stdout.read()

err=obj.stderr.read()#制作报头

header_dic={'filename':'a.txt','total_size':len(out)+len(err),'hash':'abc32i5o24'}#对包头进行序列化

header_json=json.dumps(header_dic) #字符串格式

header_bytes=header_json.encode('utf-8')#1.先发型报头的长度 len(header_bytes) struck为固定的4个字节

conn.send(struct.pack('i',len(header_bytes)))#2.发送报头

conn.send(header_bytes)#3.发送真是数据

conn.send(out)

conn.send(err)exceptConnectionResetError:breakconn.close()

server.close()

Client端配置

from socket import *

importsocket,struct,json

client=socket.socket(AF_INET,SOCK_STREAM)

client.connect(('127.0.0.1',8080))whileTrue:

cmd=input('输入你要操作的命令:')

client.send(cmd.encode('utf-8'))if len(cmd) == 0:continue

#1.先收报头的四个字节,首先拿到报头传来的长度-》bytes

header=client.recv(4) #i类型足够了 header为bytes类型

header_size=struct.unpack('i',header)[0] #拿到元祖形式,取第一个就是整个报头的长度   print(header_size)  #为报头的长度值

#2.再收报头(对应服务端的conn.send(header_bytes))

header_bytes=client.recv(header_size) #根据报头的固定长度去收接收

#3.解析包头(就是将header_bytes文件先解码成json格式)

header_str=header_bytes.decode('utf-8')

header_dic=json.loads(header_str)print(header_dic)

total_size=header_dic['total_size']print(total_size)

recv_size=0 #定义一个初始的接收变量为0,只是个计数变量,为了统计与总的total_size的len大小

res=b''

while recv_size

recv_data=client.recv(1024) #每次传过来的recv_data是bytes类型

res+=recv_data

recv_size+=len(recv_data) #循环增加每次接收值的长度#cmd=client.recv(1024)

print(res.decode('gbk'))

client.close()

粘包问题的最终解决方案,分析:

服务端:

目的为了自定义报头(报头中不仅包含长度,可能还有文件名等信息)

subprocess

...1.制作报头(字典形式)

header_dic={'filename':'a.txt','total_size':len(out)+len(err),'hash':'abc32i5o24'}2.通过json将报头序列化再encode为bytes类型

header_json=json.dumps(header_dic) #字符串类型

header_bytes=header_json.encode('utf-8') #bytes类型

3.发送报头的长度,struck目的是固定封装好的报头为4个字节长度 =====对应客户端刚开始 header=client.recv(4)

struck:1.将数字转为bytes类型,保证发送过去的是bytes类型 2.固定4个字节

第一次发:conn.send(struck.pack('i',len(header_bytes)))

先传给客户端固定了收的时候报头的长度,不至于报头和其他内容粘在一起4.发送报头 =======对应客户端接收报头 header_bytes=client.recv(header_size)

第二次发:conn.send(header_bytes)5.发送真实的数据

conn.send(out)

conn.send(err)

客户端:(bytes--int)1.通过服务端返回的字节,拿到报头的的长度

header=client.recv(4) #header是4个bytes字节

header_size=struck.unpack('i',header)[0] #字节头-拿到int大小

2.再收报头

header_bytes=client.recv(header_size) #bytes类型

3.解析报头(1.先把内存中存放的bytes类型,用decode('utf-8')解码为字符串)

header_json=header_bytes.decode('utf-8')

header_dic=json.loads(header_json) #json反序列化出字典格式 (对应server第2步)

print(header_dic)4.拿到字典中的报头大小

total_size=header_dic['total_size']print(total_size)

struck功能辅助理解:所以客户端刚开始接收4 大小是足够把报头接收完的。

#struck工鞥理解:importstruct#将整型转成bytes

res=struct.pack('i',1232435436)print(res,len(res)) #res:为bytes类型,还是固定长度4(一般情况已经可以包含很多)

#将bytes转成整型

aa=struct.unpack('i',res)print(aa,aa[0],type(aa)) #

print(len(res)) #4 一般情况多数bytes 4个长度足够了

"""结果

b'\xecxuI' 4

(1232435436,) 1232435436

4"""

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

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

相关文章

python的lib文件夹_python遍历文件夹os.path与pathlib

首先我们来一个需求,这个函数接受文件夹的名称作为输入参数,返回该文件夹中文件的路径,以及其包含文件夹中文件的路径。def print_dir_contents(sPath):import osfor sChild in os.listdir(sPath):sChildPath os.path.join(sPath,sChild)if …

python 数据字典用法_python数据字典的操作

一、什么是字典?字典是Python语言中唯一的映射类型。映射类型对象里哈希值(键,key)和指向的对象(值,value)是一对多的的关系,通常被认为是可变的哈希表。字典对象是可变的,它是一个容器类型,能存储任意个数…

双系统安装deepin20_win10deepin15.10双系统安装教程

第二步:下载深度启动盘制作工具深度启动盘制作工具地址第三步:制作U盘启动盘打开第二部下载的启动盘制作工具,并准备一个u盘插入待装系统的电脑,选择镜像文件后,下一步选择磁盘并勾选格式化磁盘,点下一步开…

ubuntu19 安装git_在Ubuntu 18.04上安装Git

步骤1.首先,通过运行以下命令确保您的系统和apt包列表完全更新:apt-get update -yapt-get upgrade -y第2步。在Ubuntu 18.04上安装Git。现在让我们安装git:apt install git您可以使用以下命令来检查已安装的git版本:$ git --versi…

mysql更新多条数据6_mysql语句:批量更新多条记录的不同值

mysql更新语句很简单,更新一条数据的某个字段,一般这样写:如果更新同一字段为同一个值,mysql也很简单,修改下where即可:这里注意 ‘other_values’ 是一个逗号(,)分隔的字符串,如&am…

php mysql query 行数_如何在PHP中获取MYSQL数据库返回的数据的行数?

展开全部1. mysql_num_rows 可得到e69da5e887aa3231313335323631343130323136353331333337383861查询记录数<?php $con mysql_connect("localhost", "hello", "321");if (!$con){die(Could not connect: . mysql_error());}$db_selected …

mysql数据库开发环境_MacOS下搭载开发环境之数据库篇(Mysql + Navicat)

一、安装Mysql1、官网下载mysql的tar包(提示&#xff1a;建议vpn环境下载)2、解压并安装tar包# 移动解压后的二进制包到安装目录sudo mv mysql-5.7.19-osx10.9-x86_64 /usr/local/mysql# 更改 mysql 安装目录所属用户与用户组cd /usr/localsudo chown -R root:wheel mysql# 初始…

mysql alter 唯一键_MySQL列属性 之 唯一键

MySQL列属性 之 唯一键唯一键唯一键&#xff1a;每张表往往有多个字段需要具有唯一性&#xff0c;数据不能重复&#xff0c;但是在每张表中&#xff0c;只能有一个主键&#xff0c;因此 唯一键就是用来解决表中多个字段需要具有唯一性的问题。例如身份证号码应该每一行的记录不…

如何在mysql中添加复选框_如何使用输入和复选框更新mysql

如果我理解正确,您需要这样做:mysql_query("UPDATE mp3SETaktif 1,baslik " . mysql_escape_string($_POST[baslik]) ."WHERE id $zuha");mysql_query("update mp3 set aktif 1,baslik $_POST[baslik]where id $_POST[id]")现在唯一的区别…

mysql 检查列是否存在,如何检查mysql表列是否存在?

How can I check if mysql table field even exists ?The column name is price and I need to see if it exists.Havent understood really how the EXISTS works...Any examples or ideas ?Thanks解决方案In PHP:$fields mysql_list_fields(database_name, table_name);$c…

mysql proxy yum_mysql 高可用架构 proxysql 之一 yum安装

os:centos 7.4mysql: 5.7proxysql: 1.4.10ip 规划如下&#xff1a;192.168.56.101 node1 (proxysql)192.168.56.102 node2 (mysql master)192.168.56.103 node3 (mysql slave)192.168.56.104 node4 (mysql slave)安装mysql 5.7node2、node3、node4 安装 mysql 5.7 software详细…

wpf 使用位图画图为什么断断续续_WPF的未来是微软WinUi!

WPF(Windows Presentation Foundation)是微软推出的基于Windows 的用户界面框架&#xff0c;属于.NET Framework 3.0的一部分。它提供了统一的编程模型、语言和框架&#xff0c;真正做到了分离界面设计人员与开发人员的工作&#xff1b;同时它提供了全新的多媒体交互用户图形界…

antd新增一行页码不正确_antd-Table@4.x对rowKey属性的重构

时间&#xff1a;2020/04/26 &#xff0c;转载请注明出处。写在前面antd团队于2020年2月发布了酝酿已久的antd4.0版本&#xff0c;对样式的调整、部分组件逻辑的重构都进行了较大改动&#xff0c;本文针对Table的rowKey属性重构作分析。由一个mistake带来的思考在数据治理模块的…

qt调用mysql调用了存储过_Qt调用Server SQL中的存储过程

Server SQL中的存储过程如下&#xff1a;CREATE procedure PINSERTPCpcnum int,pcname varchar(50),pctype int,ipaddress varchar(50),port int,pcid int outputas--declare pcid intif exists (select * from COMPUTERTABLE where PcNum pcnum)set pcid -1elsebegininser…

pandas mysql index_Pandas从入门到精通(3)- Pandas多级索引MultiIndex

首先了解一下什么是多级索引&#xff0c;以及它的作用&#xff0c;为什么要有这个玩意。多级索引也称为层次化索引(hierarchical indexing)&#xff0c;是指数据在一个轴上(行或者列)拥有多个(两个以上)索引级别。之所以引入多级索引&#xff0c;在于它可以使用户能以低维度形式…

tensorflow 启动多个session_Tensorflow源码解析7 -- TensorFlow分布式运行时

1 概述TensorFlow架构设计精巧&#xff0c;在后端运行时这一层&#xff0c;除了提供本地运行时外&#xff0c;还提供了分布式运行时。通过分布式训练&#xff0c;在多台机器上并行执行&#xff0c;大大提高了训练速度。前端用户通过session.run()启动系统执行时&#xff0c;tar…

shell swt 样式_SWT之路:SWT图像显示

简明现代魔法 -> Java编程语言 -> SWT之路&#xff1a;SWT图像显示SWT之路&#xff1a;SWT图像显示2009-10-03程序演示还是先用SWT Desiner创建界面程序。然后创建一个Display对象和Image对象&#xff0c;和一个GC对象。类org.eclipse.swt.graphics.GC是一个封装了所有可…

swool tcp mysql_swoole/mysql(异步)

# 异步Swoole\Mysql**(要求Workerman版本>3.3.6)**## 注意:此组件由swoole底层提供&#xff0c;由C语言编写&#xff0c;具有超高性能。## 安装&#xff1a;安装有swoole扩展即可## 示例&#xff1a;phprequire_once ../Autoloader.php;use Workerman\Worker;use \Swoole\My…

xamarin和mysql_Xamarin.Android 使用 SQLiteOpenHelper 进行数据库操作

一、前言在手机中进行网络连接不仅是耗时也是耗电的&#xff0c;而耗电却是致命的。所以我们就需要数据库帮助我们存储离线数据&#xff0c;以便在用户未使用网络的情况下也可以能够使用应用的部分功能&#xff0c;而在需要网络连接的功能上采用提示方式&#xff0c;让用户决定…

python 绝对值误差小于10-6_Python 被低估了的 10 个小技巧

hi&#xff0c;各位朋友们&#xff0c;小帅b回来啦&#xff0c;几日不见&#xff0c;想我了么&#xff1f;今天给大家分享几个我认为不错的 Python 小技巧&#xff0c;有些可能被你低估了哟&#xff0c;get 起来&#xff01;那么接下来就是&#xff1a;学习 Python 的正确姿势俗…