zoukankan      html  css  js  c++  java
  • 解决粘包问题

    粘包问题

    只有TCP有粘包现象,UDP不会粘包。

    123-粘包问题-粘包可能.png

    假设客户端分别发送了两个数据包D1和D2给服务端,由于服务端一次读取到的字节数是不确定的,故可能存在以下4种情况。

    1. 服务端分两次读取到了两个独立的数据包,分别是D1和D2,没有粘包和拆包;
    2. 服务端一次接收到了两个数据包,D1和D2粘合在一起,被称为TCP粘包;
    3. 服务端分两次读取到了两个数据包,第一次读取到了完整的D1包和D2包的部分内容,第二次读取到了D2包的剩余内容,这被称为TCP拆包;
    4. 服务端分两次读取到了两个数据包,第一次读取到了D1包的部分内容D1_1,第二次读取到了D1包的剩余内容D1_2和D2包的整包。

    特例:如果此时服务端TCP接收滑窗非常小,而数据包D1和D2比较大,很有可能会发生第五种可能,即服务端分多次才能将D1和D2包接收完全,期间发生多次拆包。

    粘包的两种情况

    1. 发送端需要等缓冲区满才发送出去,造成粘包(发送数据时间间隔很短,数据了很小,会合到一起,产生粘包)

    服务端

    # _*_coding:utf-8_*_
    __author__ = 'nickchen121'
    from socket import *
    ip_port = ('127.0.0.1', 8080)
    
    TCP_socket_server = socket(AF_INET, SOCK_STREAM)
    TCP_socket_server.bind(ip_port)
    TCP_socket_server.listen(5)
    
    conn, addr = TCP_socket_server.accept()
    
    data1 = conn.recv(10)
    data2 = conn.recv(10)
    
    print('----->', data1.decode('utf-8'))
    print('----->', data2.decode('utf-8'))
    
    conn.close()
    

    客户端

    # _*_coding:utf-8_*_
    __author__ = 'nickchen121'
    import socket
    BUFSIZE = 1024
    ip_port = ('127.0.0.1', 8080)
    
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    res = s.connect_ex(ip_port)
    
    s.send('hello'.encode('utf-8'))
    s.send('feng'.encode('utf-8'))
    
    1. 接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据,服务端只收了一小部分,服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包)

    服务端

    # _*_coding:utf-8_*_
    __author__ = 'nickchen121'
    from socket import *
    ip_port = ('127.0.0.1', 8080)
    
    TCP_socket_server = socket(AF_INET, SOCK_STREAM)
    TCP_socket_server.bind(ip_port)
    TCP_socket_server.listen(5)
    
    conn, addr = TCP_socket_server.accept()
    
    data1 = conn.recv(2)  # 一次没有收完整
    data2 = conn.recv(10)  # 下次收的时候,会先取旧的数据,然后取新的
    
    print('----->', data1.decode('utf-8'))
    print('----->', data2.decode('utf-8'))
    
    conn.close()
    

    客户端

    # _*_coding:utf-8_*_
    __author__ = 'nickchen121'
    import socket
    BUFSIZE = 1024
    ip_port = ('127.0.0.1', 8080)
    
    s = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    res = s.connect_ex(ip_port)
    
    s.send('hello feng'.encode('utf-8'))
    

    为何TCP可靠,UDP不可靠

    TCP在数据传输时,发送端先把数据发送到自己的缓存中,然后协议控制将缓存中的数据发往对端,对端返回一个ack=1,发送端则清理缓存中的数据,对端返回ack=0,则重新发送数据,所以TCP是可靠的

    解决粘包问题(low版本)

    问题的根源在于,接收端不知道发送端将要传送的字节流的长度,所以解决粘包的方法就是围绕,如何让发送端在发送数据前,把自己将要发送的字节流总大小让接收端知晓,然后接收端来一个死循环接收完所有数据。

    服务端

    import socket, subprocess
    
    server = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
    
    server.bind(('127.0.0.1', 8000))
    server.listen(5)
    
    while True:
        conn, addr = server.accept()
    
        print('start...')
        while True:
            cmd = conn.recv(1024)
            print('cmd:', cmd)
    
            obj = subprocess.Popen(cmd.decode('utf8'),
                                   shell=True,
                                   stderr=subprocess.PIPE,
                                   stdout=subprocess.PIPE)
    
            stdout = obj.stdout.read()
    
            if stdout:
                ret = stdout
            else:
                stderr = obj.stderr.read()
                ret = stderr
    
            ret_len = len(ret)
    
            conn.send(str(ret_len).encode('utf8'))
    
            data = conn.recv(1024).decode('utf8')
    
            if data == 'recv_ready':
                conn.sendall(ret)
    
        conn.close()
    
    server.close()
    

    客户端

    import socket
    client=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    client.connect(('127.0.0.1',8080))
    while True:
        msg=input('please enter your cmd you want>>>')
    	if len(msg)==0:continue
        client=int(client.recv(1024))
    	client.send('recv_ready'.encode('utf8'))
        
    	send_size=0
        recv_size=0
        
        data='b'
        
        while recv_size<length:
            data=client.recv(1024)
            recv_size+=len(data)
    	print(data.decode('utf8'))
    

    为何low

    程序的运行速度远快于网络传输速度,所以在发送一段字节前,先用send去发送该字节流长度,这种方式会放大网络延迟带来的性能损耗。

    借助struct模块解决

    解决粘包问题的核心就是:为字节流加上自定义固定长度报头,报头中包含字节流长度,然后一次send到对端,对端在接收时,先从缓存中取出定长的报头,然后再取真实数据。

    使用struct模块创建报头

    import json
    import struct
    
    header_dic={'filename':'a.txt',
               'total_size':21313123213123213213
               'hash':'asdf123123x123213x'}
    header_json=json.dumps(header_dic)
    
    header_bytes=header_json.encode('utf-8')
    print(len(header_bytes))
    
    #'i'是格式
    obj=struct.pack('i',len(header_bytes))
    print(obj,len(obj))
    

    87

    b'Wx00x00x00' 4

    res=struct.unpack('i',obj)
    print(res[0])
    

    87

    服务端

    from socket import *
    import subprocess
    import struct
    import json
    
    server = socket(AF_INET, SOCK_STREAM)
    server.bind(('127.0.0.1', 8000))
    server.listen(5)
    
    print('start...')
    while True:
        conn, client_addr = server.accept()
        print(conn, client_addr)
    
        while True:
            cmd = conn.recv(1024)
    
            obj = subprocess.Popen(cmd.decode('utf8'),
                                   shell=True,
                                   stderr=subprocess.PIPE,
                                   stdout=subprocess.PIPE)
    
            stderr = obj.stderr.read()
            stdout = obj.stdout.read()
    
            # 制作报头
            header_dict = {
                'filename': 'a.txt',
                'total_size': len(stdout) + len(stderr),
                'hash': 'xasf123213123'
            }
            header_json = json.dumps(header_dict)
            header_bytes = header_json.encode('utf8')
    
            # 1. 先把报头的长度len(header_bytes)打包成4个bytes,然后发送
            conn.send(struct.pack('i', len(header_bytes)))
            # 2. 发送报头
            conn.send(header_bytes)
            # 3. 发送真实的数据
            conn.send(stdout)
            conn.send(stderr)
    
        conn.close()
    
    server.close()
    

    客户端

    from socket import *
    import json
    import struct
    
    client = socket(AF_INET, SOCK_STREAM)
    client.connect(('127.0.0.1', 8000))
    
    while True:
        cmd = input('please enter your cmd you want>>>')
    
        if len(cmd) == 0: continue
    
        client.send(cmd.encode('utf8'))
    
        # 1. 先收4个字节,这4个字节中包含报头的长度
        header_len = struct.unpack('i', client.recv(4))[0]
    
        # 2. 再接收报头
        header_bytes = client.recv(header_len)
    
        # 3. 从包头中解析出想要的东西
        header_json = header_bytes.decode('utf8')
        header_dict = json.loads(header_json)
        total_size = header_dict['total_size']
    
        # 4. 再收真实的数据
        recv_size = 0
        res = b''
        while recv_size < total_size:
            data = client.recv(1024)
    
            res += data
            recv_size += len(data)
    
        print(res.decode('utf8'))
    
    client.close()
    
  • 相关阅读:
    人工智能 tensorflow框架-->简介及安装01
    【亲测】自动构建多个指定的class并发执行:Jenkins+Maven+Testng框架
    【亲测】Appium测试Android混合应用时,第二次切换到WebView失败
    appium_v1.4.16版本自动化适配android7.0系统
    python之拆包与装包
    python3之线程
    python3之进程
    python3之tcp
    python3之udp
    python3面向对象(4)之__new__方法和__init__方法
  • 原文地址:https://www.cnblogs.com/zhoajiahao/p/11116508.html
Copyright © 2011-2022 走看看