WSAEWOULDBLOCK send 出错(转)

本文解析了Winsock发送数据时遇到WSAEWOULDBLOCK异常的原因,并提出了有效的解决方案。当发送方尝试向接收方发送数据的速度超过接收方处理速度时,将触发此异常。

摘要生成于 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 做个标志判断以便于规范性。

  以前对于FD_WRITE是怎么触发的,相关书籍上虽然清楚的列出了三种条件,却也是看不太明了。在今天,自己实现了一个通信程序的第一步骤:消息交换(呵呵),用单步跟踪的方式,在服务端程序上清楚的看了一回客户端程序连接、发送数据时服务端程序的消息反应。

  我的程序中,接收的消息不多,是主要的四种:FD_ACCEPT 、FD_READ 、FD_WRITE 、FD_CLOSE。

程序开始,连接的时候,服务端接收的是FD_ACCEPT FD_WRITE,接收数据时,就只有FD_READ 了。

这个时候,再回想书上列的三种条件:

1、使用connect或WSAConnect,一个套接字首次建立了连接;

2、使用accept或WSAAccept,套接字被接受后;

3、若send、WSASend 、Sendto、或WSASendTo操作失败,返回了WSAWOULDBLOCK错误,而且缓冲的窨变得可用。


The   FD_WRITE   network   event   is   handled   slightly   differently.   An   FD_WRITE   network   event   is   recorded   when   a   socket   is   first   connected   with   connect/WSAConnect   or   accepted   with   accept/WSAAccept,   and   then   after   a   send   fails   with   WSAEWOULDBLOCK   and   buffer   space   becomes   available.   Therefore,   an   application   can   assume   that   sends   are   possible   starting   from   the   first   FD_WRITE   network   event   setting   and   lasting   until   a   send   returns   WSAEWOULDBLOCK.   After   such   a   failure   the   application   will   find   out   that   sends   are   again   possible   when   an   FD_WRITE   network   event   is   recorded   and   the   associated   event   object   is   set. 
### Winsock `send` 函数的使用说明 #### 1. 函数原型 Winsock 中的 `send` 函数用于向已连接的套接字发送数据。其函数声明如下: ```c int send( _In_ SOCKET s, _In_reads_bytes_(len) const char *buf, _In_ int len, _In_ int flags ); ``` 参数解释: - **s**: 已经建立连接的有效套接字描述符[^3]。 - **buf**: 存储要发送数据的缓冲区指针。 - **len**: 缓冲区中的字节数,表示希望发送的数据长度。 - **flags**: 发送操作的行为标志位,通常设置为0。 返回值: - 成功时返回实际发送的字节数。 - 如果发生错误,则返回 `SOCKET_ERROR` 并可通过 `WSAGetLastError()` 获取具体错误码[^5]。 --- #### 2. 示例代码 以下是使用 `send` 函数的一个简单示例: ```c #include <winsock2.h> #include <stdio.h> #pragma comment(lib,"ws2_32.lib") void SendData(SOCKET clientSocket) { const char* message = "Hello, Server!"; int bytesSent = send(clientSocket, message, strlen(message), 0); if (bytesSent == SOCKET_ERROR) { printf("Send failed with error %d\n", WSAGetLastError()); } else { printf("Bytes sent: %d\n", bytesSent); } } int main() { WORD wVersionRequested; WSADATA wsaData; wVersionRequested = MAKEWORD(2, 2); if (WSAStartup(wVersionRequested, &wsaData) != 0) { printf("WSAStartup failed.\n"); return -1; } SOCKET clientSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP); if (clientSocket == INVALID_SOCKET) { printf("Error at socket(): %ld\n", WSAGetLastError()); WSACleanup(); return -1; } struct sockaddr_in serverAddress; memset(&serverAddress, 0, sizeof(serverAddress)); serverAddress.sin_family = AF_INET; serverAddress.sin_addr.s_addr = inet_addr("127.0.0.1"); serverAddress.sin_port = htons(8080); if (connect(clientSocket, (struct sockaddr*)&serverAddress, sizeof(serverAddress)) == SOCKET_ERROR) { printf("Connect failed with error %d\n", WSAGetLastError()); closesocket(clientSocket); WSACleanup(); return -1; } SendData(clientSocket); closesocket(clientSocket); WSACleanup(); return 0; } ``` 上述代码展示了如何初始化 Winsock、创建并连接客户端套接字以及使用 `send` 函数发送消息给服务器[^4]。 --- #### 3. 常见错误及其解决方案 | 错误码 | 描述 | 解决方案 | |--------|------|----------| | `WSANOTINITIALISED` | 在调用任何 Winsock 函数前未成功调用 `WSAStartup` 初始化库[^1]。 | 调用 `WSAStartup` 完成初始化后再执行其他操作。 | | `WSAEINVAL` | 参数无效,可能是因为传递了非法的套接字或缓冲区为空。 | 检查传入参数是否合法,确保套接字有效且缓冲区不为空。 | | `WSAEWOULDBLOCK` | 当套接字处于非阻塞模式下尝试写入失败时触发此错误。 | 设置合理的等待时间或者切换到阻塞模式再重试。 | | `WSAENOTCONN` | 尝试在一个尚未完成连接过程的套接字上调用了 `send` 函数。 | 确认套接字已完成连接阶段(即 `connect` 返回正常)。 | 更多详细的错误信息可参阅官方文档获取进一步指导。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值