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()
复制代码

 

posted @   佛祖让我来巡山  阅读(342)  评论(0编辑  收藏  举报
编辑推荐:
· 从 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的设计差异
· 三行代码完成国际化适配,妙~啊~

佛祖让我来巡山博客站 - 创建于 2018-08-15

开发工程师个人站,内容主要是网站开发方面的技术文章,大部分来自学习或工作,部分来源于网络,希望对大家有所帮助。

Bootstrap中文网

点击右上角即可分享
微信分享提示