• socketserver 之 recv(1024) 问题!




    一、socket发送数据基本流程
    • 图示
      • 流程解释
        • 上述流程,为基本的收发数据流程,并且使用ack来解决粘包问题
        • 并且,最后通过获取server发送的“get file success”来给予用户友好展示

    二、实际应用问题:
    • 问题表现:
      • client已经接收了完成了所有文件数据,但且还处于阻塞状态,即:收不到"get file success"
    • 问题排查:
      • 查看 client 端收到的文件,发现 "get file success" 这个消息,被追加在了文件末尾!
      • 查看 client 端主要代码:
    with open(file, 'wb') as f1:
    recv_size = 0
    while recv_size < file_size:
    data = self.sk.recv(1024)
    recv_size += len(data)
    f1.write(data)

    print(self.sk.recv(1024).decode())  # 打印友好信息
    print('send_file:{} md5_sum:{}'.format(file_size, file_sum), 'recv_file:{} md5_sum:{}'.format(recv_size, md5_sum), sep=' ')  # 打印文件md5值
    • 原因总结:
      • 因为client是通过 sk.recv(1024)来进行接受数据,而1024表示,最多每次接受1024字节,
      • 当文件本身所有数据小于1024时候,那么,最后的"get file success"友好信息也会被client一次性接受,表现在data = self.sk.recv(1024)
      • 即此时:
        • data包涵了文件本身的所有数据+"友好信息"数据,
        • 而当代码走到print(self.sk.recv(1024).decode())时,便会阻塞(实际上server已经将友好信息发送,只不过因为1024的原因,当做了文件本身数据)
    • 解决方式:
      • 1、更改1024为更小单位,比如recv(10)
        • 接收次数变得更多,且若只要是文件,则一定有余数(文件大小 / 10),比如到最后剩余3字节没有接受完,而 3< 10,所以这3字节还是会和“友好信息”混在一起
      • 2、不发送友好信息
        • 在client端直接判断 recv_data_size 和 send_data_size,并计算文件哈希值,来判断文件是否完整,
        • 本就该如此做,即删掉两端关于“友好信息”的交互代码












  • 相关阅读:
    WPF : ListBox的几种Template属性
    提问的角度 5W1H,GRETT
    软件=科学+技术+工程, 软件的成败, 软件开发的首要任务
    软件的几个问题
    体会
    "复述一遍"是沟通的好办法
    WCF客户端调用IIS上WebService的安全问题
    C#处理chart
    js获取asp.net服务器端控件的值Demo
    SQL Server的复合索引适当用法
  • 原文地址:https://www.cnblogs.com/qiaogy/p/5845409.html
Copyright © 2020-2023  润新知