一、socket发送数据基本流程
- 图示
- 流程解释
- 上述流程,为基本的收发数据流程,并且使用ack来解决粘包问题,
- 并且,最后通过获取server发送的“get file success”来给予用户友好展示
二、实际应用问题:
- 问题表现:
- client已经接收了完成了所有文件数据,但且还处于阻塞状态,即:收不到"get file success"
- 问题排查:
- 查看 client 端收到的文件,发现 "get file success" 这个消息,被追加在了文件末尾!
- 查看 client 端主要代码:
with open(file, 'wb') as f1:recv_size = 0while 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,并计算文件哈希值,来判断文件是否完整,
- 本就该如此做,即删掉两端关于“友好信息”的交互代码