zoukankan      html  css  js  c++  java
  • socket 粘包

    产生粘包:

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

    2.接收方不及时接收缓冲区的包,造成多个包接收(客户端发送了一段数据,服务端只收了一小部分,

    服务端下次再收的时候还是从缓冲区拿上次遗留的数据,产生粘包

    所谓粘包问题主要还是因为接收方不知道消息之间的界限,不知道一次性提取多少字节的数据所造成的。

    补充:

    recv里指定的1024意思是从缓存里一次拿出1024个字节的数据

    send的字节流是先放入己端缓存,然后由协议控制将缓存内容发往对端,如果待发送的字节流大小大于缓存剩余空间,那么数据丢失,用sendall就会循环调用send,数据不会丢失

    1.解决方法:

    先发送消息的长度,再发送消息,接收的时候可以根据消息的长度循环取数据

    或者发送的数据量比较小时,可以把消息加载在一个长字符串中传递,填充满缓存

    服务端:
    #
    _*_coding:utf-8_*_ from socket import * import subprocess import struct ip_port=('127.0.0.1',8080) BUFSIZE=1024 tcp_socket_server=socket(AF_INET,SOCK_STREAM) tcp_socket_server.bind(ip_port) tcp_socket_server.listen(5) while True: conn,addr=tcp_socket_server.accept() print('客户端',addr) while True: cmd=conn.recv(BUFSIZE) print(cmd.decode('utf-8')) if len(cmd) == 0:break res=subprocess.Popen(cmd.decode('utf-8'),shell=True, #本地执行客户端传过来的命令 stdout=subprocess.PIPE, stdin=subprocess.PIPE, stderr=subprocess.PIPE) stderr=res.stderr.read() if stderr: back_msg=stderr else: back_msg=res.stdout.read() print(len(back_msg)) conn.send(struct.pack('i',len(back_msg))) #先发送长度 i 4个字节 conn.sendall(back_msg) #再发送内容 # conn.close()
    #_*_coding:utf-8_*_
    
    import socket
    import struct
    BUFSIZE=100                    #故意设小一点,一次取不完
    ip_port=('127.0.0.1',8080)
    
    s=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
    res=s.connect_ex(ip_port)
    
    while True:
        msg=input('>>: ').strip()              #输入命令
        if len(msg) == 0:continue
        if msg == 'quit':break
    
        s.sendall(msg.encode('utf-8'))
        head_info=s.recv(4)       #和服务端对应起来,pack i 是4个字节
        head_len=struct.unpack('i',head_info)[0]       #解压pack的内容
        print("head_len",head_len)
        res=0
        data=b''
    
        while res < head_len:            #循环取数据
            data+=s.recv(BUFSIZE)
            res=len(data)
            # print('res',res)
            # r_d=s.recv(BUFSIZE)
            # print('r_d',len(r_d))
            # data+=r_d
            # print('data',len(data))
            # res+=len(r_d)
    
        print(data.decode('gbk'))
  • 相关阅读:
    1.border-image
    CSS3 3D transform
    js表单的focus()与blur()方法
    jquery背景backgroundPosition插件
    数字反转
    js的字符串charAt()方法
    FormData使用方法详解
    封装自己的jquery插件
    webpack打包vue项目之后怎么启动&注意事项
    JavaScript中的async/await
  • 原文地址:https://www.cnblogs.com/wuxi9864/p/9973085.html
Copyright © 2011-2022 走看看