linux网络编程--阻塞与非阻塞
linux网络编程--阻塞与非阻塞
- 建立连接
- 接受连接
- 无阻塞的设置方式
- read() write()
- 读操作
- 写操作
- Linux fcntl函数详解
- 功能描述
- 函数原型
- fcntl()函数五种功能
- F_DUPFD
- F_GETFD
- F_SETFD
- F_GETFL
- F_SETFL
- F_GETOWN
- F_SETOWN
- F_GETFL和F_SETFL的标志
- 与上面相同,如果一个fd本身就是sockfd的时候,可以直接这是socket标志
建立连接
- 阻塞方式下,connect首先发送SYN请求道服务器,当客户端收到服务器返回的SYN的确认时,则connect返回.否则的话一直阻塞.
- 非阻塞方式,connect将启用TCP协议的三次握手,但是connect函数并不等待连接建立好才返回,而是立即返回。返回的错误码为EINPROGRESS,表示正在进行某
种过程.
接受连接
- 对于阻塞方式的倾听socket,accept在连接队列中没有建立好的连接时将阻塞,直到有可用的连接,才返回。
- 非阻塞方式倾听socket,在有没有连接时都立即返回,没有连接时,返回的错误码为EWOULDBLOCK,表示本来应该阻塞。
无阻塞的设置方式
方法一:fcntl //文件控制函数
int flag;
if (flag = fcntl(fd, F_GETFL, 0) <0)perror("get flag");
flag |= O_NONBLOCK;
if (fcntl(fd, F_SETFL, flag) < 0)perror("set flag");
方法二:ioctl //设备控制接口函数
int b_on = 1;
ioctl (fd, FIONBIO, &b_on);
read() write()
一篇文章完全搞清楚 scoket read/write 返回码、阻塞与非阻塞、异常处理 等让你头疼已久的问题
读操作
- 对于阻塞的socket,当socket的接收缓冲区中没有数据时,read调用会一直阻塞住,直到有数据到来才返回。当socket缓冲区中的数据量小于期望读取的数据量时,返回实际读取的字节数。当sockt的接收缓冲区中的数据大于期望读取的字节数时,读取期望读取的字节数,返回实际读取的长度。
- 对于非阻塞socket而言,接收缓冲区中有数据时,与阻塞socket有数据的情况是一样的,如果接收缓冲区中没有数据,则立刻返回错误号为EWOULDBLOCK,表示该操作本来应该阻塞的,但是由于本socket为非阻塞的socket,因此立刻返回,遇到这样的情况,可以在下次接着去尝试读取。如果返回值是其它负值,则表明读取错误。
非阻塞read调用方式:
if ((nread = read(sock_fd, buffer, len)) < 0)
{if (errno == EAGAIN) // EWOULDBLOCK{return 0; //表示没有读到数据}else return -1; //表示读取失败
}
else return nread;读到数据长度
写操作
- 对于阻塞Socket而言,如果发送缓冲区没有空间或者空间不足的话,write操作会直接阻塞住,如果有足够空间,则拷贝所有数据到发送缓冲区,然后返回.
- 非阻塞socket在发送缓冲区没有空间时会直接返回错误号EWOULDBLOCK,表示没有空间可写数据,如果错误号是别的值,则表明发送失败。如果发送缓冲区中有足够空间或者是不足以拷贝所有待发送数据的空间的话,则拷贝前面N个能够容纳的数据,返回实际拷贝的字节数。
非阻塞的write操作一般写法是:
int write_pos = 0;
int nLeft = nLen;while (nLeft > 0)
{int nWrite = 0;if ((nWrite = write(sock_fd, data + write_pos, nLeft)) <= 0){if (errno == EAGAIN){nWrite = 0;}else return -1; //表示写失败}nLeft -= nWrite;write_pos += nWrite;
}
return nLen;
Linux fcntl函数详解
功能描述
根据【文件描述词】来操作文件的特性。
fcntl()
针对(文件)描述符提供控制.参数fd是被参数cmd操作(如下面的描述)的描述符. 针对cmd的值,fcntl能够接受第三个参数(arg)
函数原型
#include <unistd.h>
#include <fcntl.h>int fcntl(int fd, int cmd);
int fcntl(int fd, int cmd, long arg);
int fcntl(int fd, int cmd, struct flock *lock);
fcntl()函数五种功能
- 复制一个现有的描述符(cmd=F_DUPFD).
- 获得/设置文件描述符标记(cmd=F_GETFD或F_SETFD).
- 获得/设置文件状态标记(cmd=F_GETFL或F_SETFL).
- 获得/设置异步I/O所有权(cmd=F_GETOWN或F_SETOWN).
- 获得/设置记录锁(cmd=F_GETLK,F_SETLK或F_SETLKW).
F_DUPFD
返回一个如下描述的(文件)描述符:
- 最小的大于或等于arg的一个可用的描述符
- 与原始操作符一样的某对象的引用
- 如果对象是文件(file)的话,返回一个新的描述符,这个描述符与arg共享相同的偏移量(offset)
- 相同的访问模式(读,写或读/写)
- 相同的文件状态标志(如:两个文件描述符共享相同的状态标志)
- 与新的文件描述符结合在一起的close-on-exec标志被设置成交叉式访问execve(2)的系统调用
F_GETFD
取得与文件描述符fd联合close-on-exec标志,类似FD_CLOEXEC.如果返回值和FD_CLOEXEC进行与运算结果是0的话,文件保持交叉式访问exec(),否则如果通过exec运行的话,文件将被关闭(arg被忽略)
F_SETFD
设置close-on-exec旗标。该旗标以参数arg的FD_CLOEXEC位决定。
F_GETFL
取得fd的文件状态标志,如同下面的描述一样(arg被忽略)
F_SETFL
设置给arg描述符状态标志,可以更改的几个标志是:O_APPEND, O_NONBLOCK,O_SYNC和O_ASYNC。
F_GETOWN
取得当前正在接收SIGIO或者SIGURG信号的进程id或进程组id,进程组id返回成负值(arg被忽略)
F_SETOWN
设置将接收SIGIO和SIGURG信号的进程id或进程组id,进程组id通过提供负值的arg来说明,否则,arg将被认为是进程id (《UNPV》P510)
fcntl(connfd, F_SETOWN, getpid()); // 为某个socket套接字connfd和当前线程建立属主关系,主要是为了信号SIGIO and SIGURG
F_GETFL和F_SETFL的标志
- O_NONBLOCK:非阻塞I/O;如果read(2)调用没有可读取的数据,或者如果write(2)操作将阻塞,read或write调用返回-1和EAGAIN错误
- O_APPEND:强制每次写(write)操作都添加在文件大的末尾,相当于open(2)的O_APPEND标志
- O_DIRECT:最小化或去掉reading和writing的缓存影响.系统将企图避免缓存你的读或写的数据.如果不能够避免缓存,那么它将最小化已经被缓存了的数 据造成的影响.如果这个标志用的不够好,将大大的降低性能
- O_ASYNC:当I/O可用的时候,允许SIGIO信号发送到进程组,例如:当有数据可以读的时候
- O_CLOEXEC: 当fork子进程后,仍然可以使用fd。但执行exec后系统就会字段关闭子进程中的fd了。
与上面相同,如果一个fd本身就是sockfd的时候,可以直接这是socket标志
- SOCK_NONBLOCK: 非阻塞, Using this flag saves extra calls to fcntl(2) to achieve the same result.
- SOCK_CLOEXEC:
int connfd = ::accept(sockfd, sockaddr_cast(addr),&addrlen, SOCK_NONBLOCK | SOCK_CLOEXEC);
// 或者
int sockfd = ::socket(family, SOCK_STREAM | SOCK_NONBLOCK | SOCK_CLOEXEC, IPPROTO_TCP);
注意:在修改文件描述符标志或文件状态标志时必须谨慎,先要取得现在的标志值,然后按照希望修改它,最后设置新标志值。不能只是执行F_SETFD或F_SETFL命令,这样会关闭以前设置的标志位。
linux网络编程--阻塞与非阻塞相关推荐
- unix网络编程 str_cli epoll 非阻塞版本
unix网络编程 str_cli epoll 非阻塞版本 unix网络编程str_cli使用epoll实现讲了使用epoll配合阻塞io来实现str_cli,这个版本是配合非阻塞io. 可以看到采用非 ...
- Linux网络编程 | IO模型 :阻塞IO、非阻塞IO、信号驱动IO、异步IO、多路复用IO
目录 IO模型 阻塞与非阻塞 同步与异步 阻塞IO 非阻塞IO 信号驱动IO 多路复用IO 异步IO IO模型 根据各自的特性不同,IO模型被分为阻塞IO.非阻塞IO.信号驱动IO.异步IO.多路复用 ...
- 网络编程学习笔记(非阻塞accept)
修改TCP回射客户程序,在跟服务器建立连接后发送一个RST.这种情况可以如下模拟: 一旦建立连接,设置SO_LINGER选项,把l_onoff标志置为1,l_linger时间置为0.在关闭连接时,TC ...
- 网络编程学习笔记(非阻塞connect)
设置非阻塞,如果返回EINPROGRESS,表示正在建立连接,还没有完成 非阻塞的三个用途: 1.我们可以在三路握手的同时做其它的处理.connect要花一个往返时间完成,而且可以是在任何地主,从几个 ...
- 网络编程学习笔记(非阻塞读和写)
维护两个缓冲区:to容纳从标准输入到服务器的数据,from容纳从服务器到标准输出的数据 toiptr指向从标准输入读入的数据可以存放的下一个字节,tooptr指下一个必须写入到套接口的字节.有(toi ...
- Linux Socket网络编程UDP、TCP 阻塞与非阻塞 断线重连机制
三种非阻塞模式的方法: (1) fcntl函数 int Mode = fcntl(sockfd, F_GETFL, 0); //获取文件的Mode值 fcntl(sockfd, F ...
- 【Linux网络编程学习】阻塞、非阻塞、同步、异步以及五种I/O模型
文章目录 1. 基本概念 1.1 阻塞与非阻塞 1.2 同步与异步 1.3 为什么没有"异步阻塞" 2. 五种IO模型 2.1 阻塞 blocking 2.2 非阻塞 non-bl ...
- Linux网络编程之阻塞与非阻塞
文章目录 第1关:阻塞 第2关:非阻塞 第3关:阻塞与非阻塞切换 第1关:阻塞 #include <sys/socket.h> // 包含套接字函数库 #include <fcntl ...
- C++网络编程快速入门(三):阻塞与非阻塞式调用网络通信函数
目录 阻塞与非阻塞定义 send与recv connect 一些问题 为什么要将监听socket设置为非阻塞 阻塞与非阻塞定义 阻塞模式指的是当前某个函数执行效果未达预期,该函数会阻塞当前的执行线程, ...
最新文章
- 看完让你彻底搞懂Websocket原理
- JMeter学习笔记--JMeter常用测试元件
- Anaconda中文版
- V神最新演讲:以太坊2.0的分片交易
- Dajngo-Xadmin 修改菜单摆放排序
- 导出EXCEL遇到问题
- linux操作系统之读写锁
- 增加一行减少一行_【第2171期】一行 CSS 代码的魅力
- 可折叠的 iPhone 要来了?
- windows上QT设置debugger的方法
- seo 伪原创_seo文章优化(怎么样做好伪原创文章)
- 五分钟快速了解EM算法
- 2022年下半年系统集成项目管理工程师(软考中级)报名条件
- Windows新建桌面和切换
- Structure SLAM 相关论文阅读(一):消影点/消失点/灭点检测提取
- python实现批量批量生成二维码并粘贴到另外图片上,便于打印扫描测试
- 添加打印机无法搜索计算机,添加打印机搜索不到打印机怎么办
- Android MVP开发框架
- 01炼数成金TensorFlow基本概念
- Web自定义表单工具和协同办公系统之集成(1)
热门文章
- HDU3939(毕达哥拉斯三元组的解)
- 遍历Windows系统的内核模块(源码)
- VC++ 创建Windows服务
- WebRTC Linux ADM 实现中的符号延迟加载机制
- 高级数据结构与算法 | LRU缓存机制(Least Recently Used)
- 看了这个高并发系统架构,才知道我对秒杀的误解有多深
- [一]RabbitMQ-客户端源码之ConnectionFactory
- LiveVideoStackCon 2020北京站-售票通道关闭倒计时2天
- MSU高清/极慢档Codec对比:AV1压缩效率第一 速度慢
- LeetCode——树:层次遍历、前中后序遍历