粘包现象

什么是粘包现象以及socket收发消息原理

只有tcp才会出现粘包问题
解决粘包问题的核心思想:每次都会读完收到的内容

自定义报头:

固定报头长度,报头内容为数据长度

\color{red}{struct模块 }:
该模块可以把一个类型,如数字,转成固定长度的bytes,这样就可以把数据长度转换成固定长度的bytes
用法:struct.pack('i',1111111111111)

但是之前我们了解过报头的内容并不止这一个,所以,我们可以用字典的形式作为报头,字典内容比如是文件名,md5值,数据长度,然后用json来序列化这个字典给他发送出去

这样又会产生一个新的问题,,,,接收方因为无法确定字典的长度而又会粘包

所以\color{red}{终极版解决方法 }就是:
首先通过struct发送转换成bytes的字典的长度,,然后发送报头字典,最后发送数据

练习

让我们基于tcp先制作一个远程执行命令的程序(1:执行错误命令 2:执行dir 3:执行tasklist(查看进程列表))
\color{red}{注意注意:}res.stdout.read()读出的就是GBK编码的,在接收端需要用\color{red}{GBK编码}

服务端

from socket import *
import json, struct, subprocess



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

server.listen(5)
while True:
    conn, addr = server.accept()

    while True:
        try:
            cmd = conn.recv(1024)
            res = subprocess.Popen(cmd.decode('utf-8'), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE)
            stdout = res.stdout.read()
            stderr = res.stderr.read()

            # 这是报头字典
            hand_dic={
                'file_name':'a.txt',
                'md5':'4f56ds4f65ds4f',
                'data_size':len(stdout)+len(stderr)
            }

            #字典序列化后转化为二进制
            hand_json = json.dumps(hand_dic)
            hand_bytes = hand_json.encode('utf-8')

            #发报头字典长度
            hand_size=struct.pack('i',len(hand_bytes))
            conn.send(hand_size)

            #发报头字典
            conn.send(hand_bytes)

            #发数据
            conn.send(stdout + stderr)
        except ConnectionResetError:
            break
        except ConnectionAbortedError:
            break
    conn.close()
server.close()

客户端

from socket import *
import json, struct


client=socket()
client.connect(('127.0.0.1',8080))

while True:
    cmd=input('::>').strip()
    client.send(cmd.encode('utf-8'))

    #接收报头字典长度
    hand_size=struct.unpack('i',client.recv(4))[0]

    #接收报头
    hand_bytes=client.recv(hand_size)#收到bytes格式的json
    hand_json=hand_bytes.decode('utf-8')#还原成json格式
    hand=json.loads(hand_json)#反序列化

    #数据长度
    total_size=hand['data_size']

    # 循环取值
    data_size=0#已经接收的长度
    res=b''#用来存放每次取出的数据
    while data_size<total_size:
        data=client.recv(1024) #每次取1024字节
        res+=data       #取完后存于res
        data_size+=len(data)#更新已经取出的长度
    print(res.decode('gbk'))#打印取出的数据,注意!!!Windows默认gbk编码的,所以这里要用gbk解码


client.close()

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 136,183评论 19 139
  • 目录 socket是什么? 如上图,socket(套接字)是应用层与TCP/IP协议族通信的中间软件抽象层,它是一...
    CaiGuangyin阅读 4,654评论 0 4
  • 粘包发生的场景 当应用程序使用TCP协议发送数据时,由于TCP是基于流式的数据协议,会将数据像水流一样粘在一起,当...
    断尾壁虎V阅读 5,228评论 1 1
  • 2018年7月27日星期五 2015年参加省骨干培训时初识李晓蓉校长,她的胆识、魅力、行动力、感召力就给我留下了深...
    崔承惠阅读 3,134评论 0 0
  • (一) 2016.11.26 7:30 今天起得不早不晚。 怒气还未消。 大部分人还沉浸在梦乡的时候,我已收拾好...
    码字小伙的奇葩理想阅读 3,723评论 0 0

友情链接更多精彩内容