WSAEWOULDBLOCK send 出错

本文分析了Winsock发送数据时遇到WSAEWOULDBLOCK异常的原因,并提出了有效的解决方案。当发送方输出缓冲区满或接收方处理速度慢时会出现此异常。文章建议使用FD_WRITE消息机制来解决此问题。

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

 今天有朋友问我关于 Winsock 发送数据出错的问题,错误代码为 WSAEWOULDBLOCK。而刚好以前自己也遇到过这个问题,也研究过一下发生的原因,所以很顺利的帮朋友解决了问题,但由于自己语言表达能力太弱,所以干脆把原因分析写下来:“关于 Winsock Send 无法完成,返回 WSAEWOULDBLOCK 的原因分析和解决方法”,如下:

首先,Winsock 异常 10035 WSAEWOULDBLOCK (WSAGetLastError) 的意识是 Output Buffer 已经满了,无法再写入数据。确切的说它其实不算是个错误,出现这种异常的绝大部分时候其实都不存在 Output Buffer 已满情况,而是处于一种“忙”的状态,而这种“忙”的状态还很大程度上是由于接收方造成的。意思就是你要发送的对象,对方收的没你发的快或者对方的接受缓冲区已被填满,所以就返回你一个“忙”的标志,而这时你再发多少数据都没任何意义,所以你的系统就抛出个 WSAEWOULDBLOCK 异常通知你,叫你别再瞎忙活了。

那么,我该怎么办呢?网上有很多朋友的做法是遇到这种情况就 Sleep 一段时间,一般短暂停顿后 Output Buffer 就空出来了,那就又可以继续发送了。不过我推荐另外的方法:根据 MSDN 文档所示,当出现 WSAEWOULDBLOCK 异常后直到空出 Output Buffer 时,系统会发送一个 FD_WRITE 给发送方。我们完全可以在等收到 FD_WRITE 消息后再重新发送从出现异常开始的数据包即可(该包需要全部重新发送)。

至此,该问题结案。最后顺便提一下:FD_WRITE 消息会在至少三钟情况下出现,而上面只是其中的一种,所以我建议给 Socket 做个标志判断以便于规范性。

下面的代码中,同一个函数被不同的线程调用,同一个线程共用一个socket, 是否可以按照代码中这样不断重新绑定socket 对应的local ip和端口号?为什么?DWORD WINAPI SendThreadProc(LPVOID param) { std::cout << "Start SendThread Success" << std::endl; SOCKET sock = socket(AF_INET, SOCK_DGRAM, IPPROTO_UDP); // 配置socket参数 int buf_size = 1024 * 1024; // 1MB setsockopt(sock, SOL_SOCKET, SO_SNDBUF, (char*)&buf_size, sizeof(buf_size)); u_long mode = 1; ioctlsocket(sock, FIONBIO, &mode); // 非阻塞模式 sockaddr_in dest_addr{}; dest_addr.sin_family = AF_INET; InetPton(AF_INET, remoteIP.c_str(), &dest_addr.sin_addr); sockaddr_in local_addr{}; local_addr.sin_family = AF_INET; InetPton(AF_INET, localIP.c_str(), &local_addr.sin_addr); while (true) { for (auto& item : port_queues) { Packet pkt; if (item.second.pop(pkt)) { dest_addr.sin_port = htons(item.first); local_addr.sin_port = htons(item.first); if (bind(sock, (sockaddr*)&local_addr, sizeof(local_addr)) == SOCKET_ERROR) { closesocket(sock); return; } // 非阻塞发送 int sent = sendto(sock, pkt.data.data(), pkt.data.size(), 0, (sockaddr*)&dest_addr, sizeof(dest_addr)); if (sent == SOCKET_ERROR && WSAGetLastError() != WSAEWOULDBLOCK) { // 错误处理 spdlog::error("sendto execute failed!"); } spdlog::info("send data to port {} and Length is {}.", item.first, pkt.data.size()); } } std::this_thread::sleep_for(std::chrono::microseconds(100)); } closesocket(sock); return 0; }
最新发布
03-08
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值