一.半连接数:
三次握手没有完成 称之为半连接
原因1 恶意客户端没有返回第三次握手信息
原因2 服务器没空及时处理你的请求
socket中 listen(半连接最大数量)
二.粘包问题
TCP流式协议, 数据之间没有分界, 就像水 一杯水和一杯牛奶倒在一起了!
UDP 用户数据报协议
粘包 仅发生在TCP协议中
-
发送端 发送的数据量小 并且间隔短 会粘
-
接收端 一次性读取了两次数据的内容 会粘
-
接收端 没有接收完整 剩余的内容 和下次发送的粘在一起
无论是那种情况,其根本原因在于 接收端不知道数据到底有多少
解决方案就是 提前告知接收方 数据的长度
解决方案
先发长度给对方 再发真实数据
客户端 -----------
import socket import struct client = socket.socket() client.connect(('192.168.13.72',666)) while True: try: # 首先客户端输入指令 cmd = input('请输入系统指令>>:').encode('utf-8') # 用struct的pack函数将收到的cmd指令的长度转成固定格式的字节 cmd_size_bytes = struct.pack('i',len(cmd)) # 然后先将这个固定格式的字节发过去. client.send(cmd_size_bytes) # 然后在发送cmd指令 这样就算后面跟着传别的东西发生了粘包,也不会取到后面的东西了 client.send(cmd) # 然后服务器将执行结果返回 ,同样先取四个字节.用struct的unpack函数 将数据的长度转回来 注意返回的是元组 res_bytes = client.recv(4) res_size = struct.unpack('i',res_bytes)[0] temp = b'' while True: # 然后 如果数据量大的话 一次直接接收内存会溢出,所以循环收取.循环收到的数据可以进行保存或者其他的各种操作 # 首先设置一个空二进制字符串 temp # 然后判断总的大小 减掉已接收字符串temp的长度如果大于等于1024 就直接接收1024字节 # 然后 if res_size-len(temp) >= 1024: temp += client.recv(1024) else: # else:的话 就代表数据已经快收完了 就接收总长度减掉已接受temp的长度 # 然后循环结束 temp += client.recv(res_size-len(temp)) break # 最后打印temp的结果即可 print(temp.decode('gbk')) except Exception as E: print(E) client.close() break
# 服务器-----------------
import socket import json import struct import subprocess server = socket.socket() server.bind(('192.168.13.72',666)) server.listen() while True: client, addr = server.accept() print('IP%s,端口%s已连接'%addr) while True: try: # 服务器首先接收四个字节的cmd命令的长度信息 size_bytes = client.recv(4) # 然后将这个信息 用unpack反序列化为整型的数字信息.同样返回的元组.取0号索引即可 data_size = struct.unpack('i',size_bytes)[0] # 然后用recv 读数据 将cmd的长度放进括号里 data = client.recv(data_size) # 然后用 subprocess执行系统命令 p_obj = subprocess.Popen(data.decode('utf-8'), shell=True, stdout=subprocess.PIPE, stderr=subprocess.PIPE) # 将输出管道和错误输出管道信息读出. out_info = p_obj.stdout.read() err_info = p_obj.stderr .read() # 然后把他俩加起来的长度加起来,两个字符串 因为只有一个有值另一个为空,直接不用判断了 info_size = len(out_info)+len(err_info) # 先讲长度用struct转为固定长度的字节 info_size_bytes = struct.pack('i',info_size) # 然后先把固定长度的字节发过去 client.send(info_size_bytes) # 再将两次的执行结果发过去 #这样客户端就可以直接接受多少了 client.send(out_info) client.send(err_info) except Exception as E: print('IP%s,端口%s已挂'%addr) client.close() break
#发送端
1.使用struct 将真实数据的长度转为固定的字节数据
2.发送长度数据
3.发送真实数据
接收端
1.先收长度数据 字节数固定
2.再收真实数据 真实可能很长 需要循环接收
发送端和接收端必须都处理粘包 才算真正的解决了