I/O复用——select和poll

概述

  I/O多路复用(multiplexing)的本质是通过一种机制(系统内核缓冲I/O数据),让单个进程可以监视多个文件描述符,一旦某个描述符就绪(一般是读就绪或写就绪),能够通知程序进行相应的读写操作。I/O复用的函数本身是阻塞的,他们提高程序的效率原因在于他们具有同时监听多个I/O事件的能力。

Linux中基于socket的通信本质也是一种I/O,使用socket()函数创建的套接字默认都是阻塞的,这意味着当sockets API的调用不能立即完成时,线程一直处于等待状态,直到操作完成获得结果或者超时出错。会引起阻塞的socket API分为以下四种:

  • 输入操作: recv()、recvfrom()。以阻塞套接字为参数调用该函数接收数据时,如果套接字缓冲区内没有数据可读,则调用线程在数据到来前一直睡眠。
  • 输出操作: send()、sendto()。以阻塞套接字为参数调用该函数发送数据时,如果套接字缓冲区没有可用空间,线程会一直睡眠,直到有空间。
  • 接受连接:accept()。以阻塞套接字为参数调用该函数,等待接受对方的连接请求。如果此时没有连接请求,线程就会进入睡眠状态。
  • 外出连接:connect()。对于TCP连接,客户端以阻塞套接字为参数,调用该函数向服务器发起连接。该函数在收到服务器的应答前,不会返回。这意味着TCP连接总会等待至少服务器的一次往返时间。

阻塞式I/O模型

   当进程在等待数据时,若该数据一直没有产生,则该进程将一直等待,直到等待的数据产生为止,这个过程中进程的状态是阻塞的。

 

  用户态进程调用recvfrom系统调用(udp)接收数据,当前内核中并没有准备好数据,该用户态进程将一直在此等待,不会进行其他的操作,待内核态准备好数据将数据从内核态拷贝到用户空间内存然后recvfrom返回成功的指示(或发生错误也返回,如系统调用被信号中断),此时用户态进行才解除阻塞的状态,处理收到的数据。

  用户态接收内核态数据的时候,主要有两个过程:内核态获得数据-->将数据从内核态的内存空间中复制到用户态进程的缓冲区中

非阻塞式I/O模型

  进程把一个套接字设置成非阻塞是在通知内核:当所请求的I/O操作非得把本进程投入睡眠才算完成,不要把本进程投入睡眠而是返回个错误。

  用户态进程调用recvfrom接收数据,当前并没有数据报文产生,此时recvfrom返回EWOULDBLOCK,用户态进程会一直调用recvfrom询问内核,待内核准备好数据的时候,之后用户态进程不再询问内核,待数据从内核复制到用户空间,recvfrom成功返回,用户态进程开始处理数据。当数据从内核复制到用户空间中的这一段时间中,用户态进程是处于阻塞的状态的。

I/O复用模型

  IO多路复用是指内核一旦发现进程指定的一个或者多个IO条件准备读取,它就通知该进程。IO多路复用适用如下场合:

  1. 当客户处理多个描述字时(一般是交互式输入和网络套接口),必须使用I/O复用。
  2. 当一个客户同时处理多个套接口时,而这种情况是可能的,但很少出现。
  3. 如果一个TCP服务器既要处理监听套接口,又要处理已连接套接口,一般也要用到I/O复用。
  4. 如果一个服务器即要处理TCP,又要处理UDP,一般要使用I/O复用。
  5. 如果一个服务器要处理多个服务或多个协议,一般要使用I/O复用。

  与多进程和多线程技术相比,I/O多路复用技术的最大优势是系统开销小,系统不必创建进程/线程,也不必维护这些进程/线程,从而大大减小了系统的开销。

  如果一个进程需要等到多种不同的消息,那么一般的做法就是开启多条线程,每个线程接收一类消息,如果每个线程都是采用阻塞式I/O模型,那么每个线程在消息未产生的时候就会阻塞,也就是说在多线程中使用阻塞式I/O。I/O复用就是基于上述的场景中,无需采用多线程监听消息的方式,进程直接监听所有的消息类型,这其中就涉及到select、poll、epoll等不同的方法。

  可以将select复用机制看作是一个描述符集合的管理,进程通过向这个集合中放入不同的描述符,用来等待不同的消息产生,然后通过select统一的进行管理,让其可以同时等待这个集合中任意一个事件的产生。I/O复用和阻塞式I/O很相似,不同的是

  1. I/O复用等待多类事件,阻塞式I/O只等待一类事件
  2. 在I/O复用中,会产生两个系统调用(select和recvfrom),而阻塞式I/O只产生一个系统调用

  那么这就涉及到具体的性能问题,当只存在一类事件的时候,使用阻塞式I/O模型的性能会更好,当存在多种不同类型的事件时,I/O复用的性能要好的多,因为阻塞式I/O模型只能监听一类事件,所以这个时候需要使用多线程进行处理

信号驱动式I/O模型

  与阻塞式和非阻塞式有了一个本质的区别,那就是用户态进程不再等待内核态的数据准备好,直接可以去做别的事情。

  先开启套接字的信号驱动I/O功能,并通过sigaction系统调用安装一个信号处理函数,该系统调用立即返回,进程继续工作,没有被阻塞。而当内核态中的数据准备好之后,内核立马发给用户态一个信号,用户态进程收到之后,立马调用在信号处理函数中调用recvfrom,等待数据从内核空间复制到用户空间(这段时间用户态是阻塞的),待完成之后recvfrom返回成功指示,用户态进程才处理别的事情。

异步I/O模型

  先用户态进程告诉内核态需要什么数据(上图中通过aio_read),然后用户态进程就不管了,并让内核在整个操作(内核等待用户态需要的数据准备好,然后将数据复制到用户空间),完成后通知我们,与信号驱动I/O区别是:信号驱动I/O通知我们何时执行一个I/O操作,而异步I/O模型是由内核通知我们I/O操作何时完成

I/O模型比较

  五种不同的I/O模型中,前四种主要区别在第一阶段,因为第二阶段是一样:从内核复制数据到调用者的缓冲区期间,进程阻塞于系统调用。

select

#include <sys/select.h>
#include <sys/time.h>

int select(int max_fd, fd_set *readset, fd_set *writeset, fd_set *exceptset, struct timeval *timeout)
  1. max_fd指定被监听的文件描述符的总数,设置为监听的所有文件描述符中最大值加1,因为文件描述符是从0开始计数的。 
  2. readfds、writefds和exceptfds参数分别指向可读、可写、异常事件对应的文件描述符集合。程序员通过这3个参数向该调用传入自己感兴趣的文件描述符。函数返回时,内核将修改它们来通知应用程序哪些文件描述符已经就绪,描述符集内任何未就绪的描述符对应的位返回时均清0,每次重新调用select时,得再次把关心的位设为1
  3. 每次返回用户注册的整个事件的集合,包括就绪的和未就绪的,所以索引就绪的文件描述符集合为O(n)
#define XFD_SETSIZE     256
#define FD_SETSIZE      XFD_SETSIZE

typedef long fd_mask;
#define NBBY    8
#define NFDBITS (sizeof(fd_mask) * NBBY)

#define howmany(x,y)    (((x)+((y)-1))/(y))

#if defined(BSD) && BSD < 198911
typedef struct fd_set 
{
    fd_mask fds_bits[howmany(FD_SETSIZE, NFDBITS)];
} fd_set;
#endif

  fd_set结构体仅包含一个long型数组,根据推导可见该数组为,它仅仅是一个文件描述符集合,没有将文件描述符和事件绑定,因此需要提供三种这样的类型的集合来分别访问输入输出和异常,不能处理更多类型的事件,

long fds_bits[8];

  fds_bit共占据32字节,即256位。该数组的每个元素的每一位(bit)标记一个文件描述符。fd_set能容纳的文件描述符由FD_SETSIZE指定,显然这限制了select()能同时处理的文件描述符的总数。该系统调用还提供了一些列宏方便程序员实现位操作:

void FD_CLR(int fd, fd_set *set); //清零set中的fd位 
int FD_ISSET(int fd, fd_set *set); //测试set中的fd位是否被设置
void FD_SET(int fd, fd_set *set); //设置fd中的fd位 
void FD_ZERO(fd_set *set); //清零set中所有位

  3.timeout参数设置函数的超时时间。它是一个timeval的普通(非const)指针,内核可以修改此参数以告诉应用程序函数阻塞等待了多久。不过内核返回的该值不能完全信任,比如调用失败时timeout的值是不确定的。timeval结构体定义如下:

struct timeval 
{
long tv_sec; /* seconds */ long tv_usec; /* microseconds */ };

  此参数有三种可能:

  1. 永远等下去:尽在有一个描述符准备好I/O才返回,此时设为NULL
  2. 等待一段固定的时间:有一个描述符准备好时返回I/O,不超过指定的秒数和微妙数
  3. 不等待立即返回:此参数必须指向一个timeval结构体,该结构体中的秒数和微妙数必须为0
socket文件描述符就绪条件

  描述符可读的依据是: 
  (1) socket内核接收缓冲区中字节数>=其低水位标记SO_RCVLOWAT时,此时程序可以无阻塞地读该socket,返回读取到的字节数(>0) 
  (2) socket通信的对端关闭连接,此时对该socket的读操作将返回0表示对端关闭 
  (3) 监听socket上有新的连接请求 
  (4) socket上有未处理的错误,此时可以使用getsockopt()读取和清除该错误(使用SO_ERROR标记)

  socket文件描述符可写: 
  (1) socket内核发送缓冲区的空闲区域大于或等于其低水位标记SO_SNDLOWAT,此时程序可以无阻塞的写该socket,返回写入的字节数(>0) 
  (2) socket的写操作被关闭(使用shotdown(fd, SHUT_WR))后再对socket写,会触发一个SIGPIPE信号 
  (3) socket使用非阻塞connect()连接成功或者失败(超时)之后,对于后者将会收到RST报文段,若收到RST报文段后继续往该socket写则会触发SIGPIPE信号 
  (4) socket上未处理的错误

注意:

  当某个套接字发生错误时,他将由select标记变为即可读又可写;接受低水位标记和发送低水位标记目的:允许应用进程控制在select返回即可读又可写条件之前有多少数据可读或有多大空间可写,例:如果我们知道除非存在64字节的数据,否则我们的应用程序没有任何有效的工作,那么可以把接受低水位标记设置为64,防止少于64字节的数据唤醒我们。

  select不知道stdio使用了缓冲区,他只是从read系统调用的角度指出是否有数据可读,而不是从fgets角度之类的考虑,所以混合使用stdio和select容易出错

 poll

  poll的机制与select类似,处理流设备时能提供额外的信息,管理多个描述符也是进行轮询,根据描述符的状态进行处理,但最大文件描述符限制为65535;

#include <poll.h>
int poll(struct pollfd fds[], nfds_t nfds, int timeout);

typedef struct pollfd
{
        int fd;                         // 需要被检测或选择的文件描述符
        short events;                   // 对文件描述符fd上感兴趣的事件
        short revents;                  // 文件描述符fd上当前实际发生的事件,内核每次修改此结构体。events不变,所以无需反复从用户空间读入这些事件*/
} pollfd_t;
/*
1.函数返回fds集合中就绪的读、写,或出错的描述符数量,返回0表示超时,返回-1表示出错;
2.fds:用于存放需要检测其状态的socket描述符,并且调用poll函数之后fds数组不会被清空;一个pollfd结构体表示一个被监视的文件
    描述符,通过传递fds[]指示 poll() 监视的文件描述符。events域是监视该文件描述符的事件掩码,由用户来设置。revents是文件
    描述符的操作结果事件掩码,内核在调用返回时设置这个域。events域中请求的任何事件都可能在revents域中返回。
3.nfds:最多监听的文件描述符数
4.timeout:是调用poll函数阻塞的超时时间,单位毫秒;
*/

   当不关心某个特定的描述符时,把它对应的结构体fd设置为负值,这样poll将忽略结构体中的events成员,返回时将revents成员设置为0

  fd成员指定文件描述符。event成员告诉内核要监听fd上的哪些事件,它可以是一系列事件的按位或。revents成员由内核修改,以通知应用程序fd上实际发生了哪些事件。event的取值为: 
这里写图片描述

  上面事件选项中: 
  a. POLLRDNORM(普通数据可读)、POLLRDBAND(优先级带数据可读)和POLLWRNORM(普通数据可写)、POLLWRBAND(优先级带数据可写)将POLLIN(数据可读)和POLLOUT(数据可写)划分得更明显,以区分优先级带数据和普通数据,但是Linux并不完全支持。 
  b. 一般应用程序调用recv()时,要判断接收到的是有效数据还是对端关闭连接后触发的是根据recv()的返回值(如上面的select()示例程序),在poll()系统调用中,有更直接的方法,监听描述符的POLLRDHUP事件即监听对端关闭事件,不过需要在代码开始处定义”_GNU_SOURCE”

  (2) fds数组成员的的个数由参数nfds指定(typedef unsigned long int nfds_t;)。显然,这个比select()的设计要灵活一点: 用户可以监测任意多数目文件描述符,但是poll()的实现也是依靠轮询的,从效率上来讲跟select()的实现是一致的。

  (3) timeout参数指定函数的超时事件,单位为毫秒。当timeout为-1时,poll调用将一直阻塞直到监听的目标事件发生;当timeout为0时,poll()调用立即返回。

  (4) poll()的返回值跟select()的返回值含义相同。

  当poll调用之后检测某事件是否发生时,fds[i].revents & POLLIN进行判断。

注意:

  1. 所有正规的TCP和UDP都是普通数据
  2. TCP的外带数据是优先数据
  3. TCP的读半部分关闭时(如收到对端的FIN)也是普通数据,随后的读操作返回0
  4. TCP连接存在错误即可认为是普通数据也可认为是错误,无论哪种情况随后的读操作都返回-1,并设置errno,这适用于处理RST或超时等待条件
  5. 监听套接字上有新的数据即可认为是普通数据也可认为是优先数据, 大多数认为是普通数据
  6. 非阻塞的connect的完成被认为是相应的套接字可写

 总结

  阻塞I/O,I/O复用和信号驱动I/O都是同步I/O,因为在这三种I/O模型中,I/O的读写操作都是在I/O事件发生后,由应用程序来完成;对于异步I/O,用户可以直接对I/O进行读写操作,这些操作告诉内核用户读写缓冲区的位置,以及I/O操作完成之后内核通知应用进程的方式,异步I/O操作总是立即返回,不论I/O是否阻塞,因为真正的读写操作已经由内核接管。即:同步I/O模型要求用户代码自行执行I/O操作(将数据从内核缓冲区读入用户缓冲区或从用户缓冲区写入内核缓冲区),异步I/O是由内核来执行I/O操作(数据在内核缓冲区和用户缓冲区的移动是由内核在“后台”完成的)。同步I/O是应用程序通知I/O就绪事件,异步I/O是通知应用程序I/O完成事件。

作用:

  1. 非阻塞的connect
  2. 同时处理udp和tcp:1>一个socket只能监听一个端口,服务器如果要同时监听多个端口,就要创建多个socket,分别将他们绑定到多个socket上,此时服务器需要监听多个socket,就可以用I/O复用;2>一个端口同时处理该端口的tcp和udp请求——创建两个不同的socket,两个不同的流都绑定到相同的端口上,就可同时处理同一端口上的tcp和udp请求。
  3. 客户端同时首发数据

转载于:https://www.cnblogs.com/tianzeng/p/9397048.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值