tcp套接字粘包解决办法
粘包只会出现在tcp,udp传输不会产生粘包现象。解决粘包的原理就是服务器预先向客户端发送客户端即将获取文件的大小。
第一版解决方案:
服务器:
1 # Author : Kelvin 2 # Date : 2019/2/2 17:38 3 from socket import * 4 import subprocess 5 6 ip_conf = ("127.0.0.1", 8888) 7 buffer_capacity = 1024 8 tcp_server = socket(AF_INET, SOCK_STREAM) 9 tcp_server.bind(ip_conf) 10 tcp_server.listen(5) 11 while True: 12 conn, addr = tcp_server.accept() 13 while True: 14 try: 15 cmd = conn.recv(buffer_capacity) # 如果强制断开连接会触发try,try正是解决强制中断连接的问题 16 print("收到的cmd:%s" % cmd) 17 if not cmd: # 如果使用quit断开连接,服务器会死循环收到空,该判断正是解决此问题 18 break 19 res = subprocess.Popen(cmd.decode("utf8"), shell=True, stdout=subprocess.PIPE, stdin=subprocess.PIPE, 20 stderr=subprocess.PIPE) 21 err = res.stderr.read() 22 if err: 23 back_msg = err 24 else: 25 back_msg = res.stdout.read() 26 if not back_msg: 27 back_msg = "acute successful!".encode("gbk") 28 length = len(back_msg) 29 conn.send(str(length).encode("gbk")) 30 re_ready = conn.recv(buffer_capacity).decode("utf8") 31 if re_ready == "ready": 32 conn.send(back_msg) 33 except Exception as e: 34 print(e) 35 break 36 tcp_server.close()
客户端:
1 # Author : Kelvin 2 # Date : 2019/2/2 17:38 3 from socket import * 4 5 ip_conf = ("127.0.0.1", 8888) 6 buffer_capacity = 1024 7 tcp_client = socket(AF_INET, SOCK_STREAM) 8 tcp_client.connect(ip_conf) 9 while True: 10 cmd = input("Please input cmd : ") 11 if not cmd: 12 continue 13 if cmd == "quit": 14 break 15 tcp_client.send(cmd.encode("utf8")) 16 re_size = int(tcp_client.recv(buffer_capacity).decode("utf-8")) 17 print("大小:", re_size) 18 tcp_client.send("ready".encode("utf8")) 19 recved_size = 0 20 recved_data = b"" 21 while recved_size < re_size: 22 recved_data += tcp_client.recv(buffer_capacity) 23 recved_size = len(recved_data) 24 back_msg = recved_data.decode("gbk") 25 print(back_msg) 26 tcp_client.close()
升级版:
服务器:
1 # Author : Kelvin 2 # Date : 2019/2/2 17:38 3 from socket import * 4 import subprocess 5 import struct 6 7 ip_conf = ("127.0.0.1", 8888) 8 buffer_capacity = 1024 9 tcp_server = socket(AF_INET, SOCK_STREAM) 10 tcp_server.bind(ip_conf) 11 tcp_server.listen(5) 12 while True: 13 conn, addr = tcp_server.accept() 14 while True: 15 try: 16 cmd = conn.recv(buffer_capacity) # 如果强制断开连接会触发try,try正是解决强制中断连接的问题 17 print("收到的cmd:%s" % cmd) 18 if not cmd: # 如果使用quit断开连接,服务器会死循环收到空,该判断正是解决此问题 19 break 20 res = subprocess.Popen(cmd.decode("utf8"), shell=True, stdout=subprocess.PIPE, stdin=subprocess.PIPE, 21 stderr=subprocess.PIPE) 22 err = res.stderr.read() 23 if err: 24 back_msg = err 25 else: 26 back_msg = res.stdout.read() 27 if not back_msg: 28 back_msg = "acute successful!".encode("gbk") 29 length = len(back_msg) 30 re_length = struct.pack("i", length) 31 conn.send(re_length) 32 conn.send(back_msg) 33 except Exception as e: 34 print(e) 35 break 36 tcp_server.close()
客户端:
1 # Author : Kelvin 2 # Date : 2019/2/2 17:38 3 from socket import * 4 import struct 5 ip_conf = ("127.0.0.1", 8888) 6 buffer_capacity = 1024 7 tcp_client = socket(AF_INET, SOCK_STREAM) 8 tcp_client.connect(ip_conf) 9 while True: 10 cmd = input("Please input cmd : ") 11 if not cmd: 12 continue 13 if cmd == "quit": 14 break 15 tcp_client.send(cmd.encode("utf8")) 16 re_size = struct.unpack("i",tcp_client.recv(4))[0] 17 print("大小:", re_size) 18 recved_size = 0 19 recved_data = b"" 20 while recved_size < re_size: 21 recved_data += tcp_client.recv(buffer_capacity) 22 recved_size = len(recved_data) 23 back_msg = recved_data.decode("gbk") 24 print(back_msg) 25 tcp_client.close()
分类:
[编程语言] Python
标签:
Python
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· 开发者必知的日志记录最佳实践
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· Manus的开源复刻OpenManus初探
· AI 智能体引爆开源社区「GitHub 热点速览」
· 从HTTP原因短语缺失研究HTTP/2和HTTP/3的设计差异
· 三行代码完成国际化适配,妙~啊~