首页 > 后端开发 > C++ > 正文

c++怎么实现非阻塞socket_c++非阻塞socket实现方法

下次还敢
发布: 2025-10-11 17:11:01
原创
904人浏览过
非阻塞Socket通过设置非阻塞模式避免I/O操作阻塞主线程,适用于高并发场景。1. 创建Socket后,Windows使用ioctlsocket(FIONBIO),Linux使用fcntl(O_NONBLOCK)设为非阻塞;2. 调用connect后若返回错误码为WSAEWOULDBLOCK(Windows)或EINPROGRESS(Linux),表示连接异步进行,需通过select检测可写事件判断是否连接成功;3. send/recv在非阻塞模式下可能仅发送或接收部分数据,send返回-1且错误码为WSAEWOULDBLOCK/EWOULDBLOCK时表示缓冲区满,应稍后重试,recv返回0表示对端关闭连接;4. 使用select可同时监控多个Socket的读、写、异常事件,实现事件驱动的多路复用。核心在于通过状态轮询而非阻塞等待完成I/O操作,跨平台开发需封装系统调用差异。

c++怎么实现非阻塞socket_c++非阻塞socket实现方法

在C++中实现非阻塞Socket,核心是将Socket设置为非阻塞模式,这样在网络I/O操作(如connect、send、recv)时不会导致程序挂起。适用于需要同时处理多个连接或避免主线程卡顿的场景。

1. 设置Socket为非阻塞模式

在创建Socket后,需通过系统调用将其设为非阻塞。不同平台方式略有差异:

Windows:
使用 ioctlsocket 函数:

#include <winsock2.h>
...
SOCKET sock = socket(AF_INET, SOCK_STREAM, 0);
unsigned long nonBlocking = 1;
ioctlsocket(sock, FIONBIO, &nonBlocking); // 设为非阻塞
登录后复制

Linux/Unix:
使用 fcntl 函数:

#include <fcntl.h>
...
int sock = socket(AF_INET, SOCK_STREAM, 0);
int flags = fcntl(sock, F_GETFL, 0);
fcntl(sock, F_SETFL, flags | O_NONBLOCK); // 添加非阻塞标志
登录后复制

2. 非阻塞connect操作

调用 connect 后,连接可能未立即完成。需通过 select 或 poll 检测是否可写,判断连接成功与否。

示例流程:

立即学习C++免费学习笔记(深入)”;

  • 调用 connect(),若返回 -1/ SOCKET_ERROR,检查错误码
  • Windows 下若错误码为 WSAEWOULDBLOCK,表示正在连接
  • Linux 下若 errno 为 EINPROGRESS,表示异步进行中
  • 使用 select 检查Socket是否可写:可写且无错误 → 连接成功

3. 非阻塞send和recv

发送与接收数据时,非阻塞Socket会立即返回结果,可能只发送/接收部分数据。

ViiTor实时翻译
ViiTor实时翻译

AI实时多语言翻译专家!强大的语音识别、AR翻译功能。

ViiTor实时翻译 116
查看详情 ViiTor实时翻译

常见处理方式:

  • send 返回 >0:实际发送字节数;返回 -1:检查错误
  • Linux 下 errno == EAGAIN 或 EWOULDBLOCK 表示当前不可写,稍后重试
  • Windows 下 WSAGetLastError() == WSAEWOULDBLOCK 同样表示缓冲区满
  • recv 返回 >0:收到数据;返回 0:对方关闭连接;返回 -1:检查是否为“假错”

建议配合 select/poll/epoll(Linux)或 IOCP(Windows)做事件驱动管理。

4. 使用select监控Socket状态

select 可同时监听多个Socket的读、写、异常事件,适合轻量级并发。

基本用法:

fd_set writeSet, exceptSet;
FD_ZERO(&writeSet);
FD_ZERO(&exceptSet);
FD_SET(sock, &writeSet);
FD_SET(sock, &exceptSet);

struct timeval timeout = {5, 0};
int result = select(0, nullptr, &writeSet, &exceptSet, &timeout);

if (result > 0) {
    if (FD_ISSET(sock, &writeSet)) {
        // 连接成功或可以发送数据
    }
    if (FD_ISSET(sock, &exceptSet)) {
        // 连接失败
    }
}
登录后复制

基本上就这些。非阻塞Socket的关键是不依赖单次调用完成操作,而是通过状态检测和事件循环处理I/O。跨平台项目建议封装统一接口,区分Windows与Unix-like系统调用差异。

以上就是c++++怎么实现非阻塞socket_c++非阻塞socket实现方法的详细内容,更多请关注php中文网其它相关文章!

c++速学教程(入门到精通)
c++速学教程(入门到精通)

c++怎么学习?c++怎么入门?c++在哪学?c++怎么学才快?不用担心,这里为大家提供了c++速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号