recv非阻塞接收的处理

非阻塞recv返回值分析,包括ret > 0的正常情况及ret == 0(连接断开)与ret == -1(错误)的特殊情况。对于ret == -1,需要考虑errno为EAGAIN和EINTR时的特殊处理,而ret == 0表示连接已断开,且不会改变errno。错误处理代码中,ret = 0和ret = -1需分开处理,以避免误判。

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

recv非阻塞接收,recv返回的ret通常有3种情况,ret = 0 / ret < 0(即-1) /ret > 0

ret > 0 是正常接收到字节数
关键是ret == 0 和 ret == -1这两种情况要不要分开写还是统一考虑

需要说明的是,当ret = -1时,说明发生了错误,错误号保存在errno中,通过errno可以分析出错误原因。但是,对于errno为EAGAIN (EWOULDBLOCK) 和 EINTR 的这两个情况,是不需要处理的,继续调用recv。除了这两种情况的其他errno,都是要处理的。
同样的 ret = 0,说明connection disconnected。这种情况是要处理的。但是此时的errno并没有被设置,ret = 0是不会修改errno的值的。即errno还是上一次的errno
于是有如下代码

ret = ::recv(socketfd, buff_ + recv_len, BUFFER_SIZE - recv_len, MSG_DONTWAIT);
if (ret == -1 || ret == 0)
{
	if (errno != EAGAIN && errno != EWOULDBLOCK && errno != EINTR)
	{
		// 处理错误
	}
	// 结束不处理错误
}
else
{
	// 处理ret > 0
}

上面这段代码是有问题的
先说明一下结论
ret = 0 和 ret = -1要分开写

if (ret == 0)
{
	// 必须处理错误
}
else if (ret == -1)
{
	if (errno != EAGAIN && errno != EWOULDBLOCK && errno != EINTR)
	{
		// 处理错误
	}
	// 结束不处理错误
}
else
{
	// 处理 ret > 0
}

对比一下。有一种情况是发生过errno为EAGAIN(很常见,因为是非阻塞recv)。后来又发生ret =0,errno就是上一次的EAGAIN。第一份代码中就会直接不处理错误了。
但是对于阻塞的recv就不需要判断这个,因为阻塞的recv是不会发生EAGAIN。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值