socket在发送一段时间数据后send卡死问题的解决

本文介绍了一种服务端在压力测试中出现send卡死现象的问题排查与解决过程。通过调整socket缓冲区大小并改进客户端recv处理,有效解决了问题。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

问题描述:

近期在对新做的服务器做压力测试时,发现服务端在接收一阵子数据之后便不再响应,不能接收新的连接,也不能接收新的数据。并且服务端接收到的数据个数非常有规律,每次都基本固定。


架构:

服务端基于select模型,测试客户端为socket client


问题查找:

1、最开始在没有跟踪代码的情况下,先是怀疑服务端在收完数据后在处理过程中,由于某种原因产生了死锁,检查代码后排除

2、程序中加入详细日志,观察程序是死在了哪个地方。多次测试后发现,服务端在回复客户端多次后,在send时卡住了

3、思考send卡住的原因:会不会跟socket的缓冲区有关系?或者是有其它配置问题?


调试解决:

1、根据资料显示,socket默认的发送缓冲大小为8192字节,使用setsockopt修改缓冲区后,服务端收到数据的次数有明显变化

	int sbuflen = 8192*2;
	setsockopt(ListenSocket, SOL_SOCKET, SO_SNDBUF, (const char*)&sbuflen, sizeof(int));
2、经分析,是由于测试客户端没有处理recv,服务端不断send,导致send缓冲区变满,从而卡死了send

3、测试客户端加入对recv的处理后,问题解决


结论:

1、资料显示socket的send缓冲区默认为8192字节(8k),但是在实际测试中发现,send的数据在达到约23k时才会卡死,为什么两个值会有这么大的差异,还需进一步核实

2、不能单纯的把缓冲区改大来解决该问题,要清楚为什么会满

3、在socket编程中,特别是基于TCP的通信,要养成“一问一答”的好习惯,在保证通信数据完整性的同时,也避免了一些出问题的可能


import cv2 import socket import numpy as np # 创建socket连接 client_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) # 设置服务器地址和端口号 server_address = ('10.132.11.225', 8000) # 连接服务器 try: client_socket.connect(server_address) except socket.error as e: print("Failed to connect to server: {}".format(e)) exit() # 采集摄像头中的照片 cap = cv2.VideoCapture(0) # 检查摄像头是否打开成功 if not cap.isOpened(): print("Failed to open camera") exit() # 设置采集频率 wait_time = 60 while True: # 等待指定时间,控制采集频率 key = cv2.waitKey(wait_time) if key == 27: # 按下ESC键退出循环 break elif key == ord(' '): # 按下空格键拍摄照片 # 采集一帧照片 ret, frame = cap.read() # 检查照片采集是否成功 if not ret: print("Failed to capture frame") break # 在窗口中显示照片 cv2.imshow('frame', frame) # 将照片转换为JPEG格式,并压缩 _, jpeg = cv2.imencode('.jpg', frame, [int(cv2.IMWRITE_JPEG_QUALITY), 70]) # 判断数据大小是否符合要求 if len(jpeg) > 1024 * 1024: print("Data size too large") continue # 将JPEG数据转换为numpy数组,发送服务器 data = np.array(jpeg).tobytes() # 发送数据服务器 try: client_socket.sendall(data) # 接收确认信息 response = client_socket.recv(1024) if response != b'OK': print("Failed to receive response from server") break except socket.error as e: print("Failed to send data to server: {}".format(e)) break # 关闭socket连接和摄像头 client_socket.close() cap.release() cv2.destroyAllWindows()优化一下
06-11
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值