select、poll、epoll之间的区别

转载自select、poll、epoll之间的区别(搜狗面试)

(1) s e l e c t select select==>时间复杂度 O ( n ) O(n) O(n)

它仅仅知道了,有 I / O I/O I/O事件发生了,却并不知道是哪几个流(可能有一个,多个,甚至全部),我们只能无差别轮询所有流,找出能读出数据,或者写入数据的流,对他们进行操作。所以 s e l e c t select select具有 O ( n ) O(n) O(n)的无差别轮询复杂度,同时处理的流越多,无差别轮询时间就越长。

(2) p o l l poll poll==>时间复杂度 O ( n ) O(n) O(n)

p o l l poll poll本质上和 s e l e c t select select没有区别,它将用户传入的数组拷贝到内核空间,然后查询每个 f d fd fd对应的设备状态, 但是它没有最大连接数的限制,原因是它是基于链表来存储的.

(3) e p o l l epoll epoll==>时间复杂度 O ( 1 ) O(1) O(1)

e p o l l epoll epoll可以理解为 e v e n t event event p o l l poll poll,不同于忙轮询和无差别轮询, e p o l l epoll epoll会把哪个流发生了怎样的 I / O I/O I/O事件通知我们。所以我们说 e p o l l epoll epoll实际上是事件驱动(每个事件关联上 f d fd fd)的,此时我们对这些流的操作都是有意义的。(复杂度降低到了 O ( 1 ) O(1) O(1)

s e l e c t select select p o l l poll poll e p o l l epoll epoll都是 I O IO IO多路复用的机制。 I / O I/O I/O多路复用就通过一种机制,可以监视多个描述符,一旦某个描述符就绪(一般是读就绪或者写就绪),能够通知程序进行相应的读写操作。但 s e l e c t select select p o l l poll poll e p o l l epoll epoll本质上都是同步 I / O I/O I/O,因为他们都需要在读写事件就绪后自己负责进行读写,也就是说这个读写过程是阻塞的,而异步 I / O I/O I/O则无需自己负责进行读写,异步 I / O I/O I/O的实现会负责把数据从内核拷贝到用户空间。

e p o l l epoll epoll s e l e c t select select都能提供多路 I / O I/O I/O复用的解决方案。在现在的 L i n u x Linux Linux内核里有都能够支持,其中 e p o l l epoll epoll L i n u x Linux Linux所特有,而 s e l e c t select select则应该是 P O S I X POSIX POSIX所规定,一般操作系统均有实现

(1) s e l e c t select select

s e l e c t select select本质上是通过设置或者检查存放 f d fd fd标志位的数据结构来进行下一步处理。这样所带来的缺点是:

1、 单个进程可监视的 f d fd fd数量被限制,即能监听端口的大小有限。

  • 一般来说这个数目和系统内存关系很大,具体数目可以 c a t cat cat / p r o c / s y s / f s / f i l e /proc/sys/fs/file /proc/sys/fs/file- m a x max max察看。32位机默认是1024个。64位机默认是2048.

2、 对 s o c k e t socket socket进行扫描时是线性扫描,即采用轮询的方法,效率较低:

  • 当套接字比较多的时候,每次 s e l e c t ( ) select() select()都要通过遍历 F D FD FD_ S E T S I Z E SETSIZE SETSIZE S o c k e t Socket Socket来完成调度,不管哪个 S o c k e t Socket Socket是活跃的,都遍历一遍。这会浪费很多 C P U CPU CPU时间。如果能给套接字注册某个回调函数,当他们活跃时,自动完成相关操作,那就避免了轮询,这正是 e p o l l epoll epoll k q u e u e kqueue kqueue做的。

3、需要维护一个用来存放大量 f d fd fd的数据结构,这样会使得用户空间和内核空间在传递该结构时复制开销大

(2) p o l l poll poll

p o l l poll poll本质上和 s e l e c t select select没有区别,它将用户传入的数组拷贝到内核空间,然后查询每个 f d fd fd对应的设备状态,如果设备就绪则在设备等待队列中加入一项并继续遍历,如果遍历完所有 f d fd fd后没有发现就绪设备,则挂起当前进程,直到设备就绪或者主动超时,被唤醒后它又要再次遍历 f d fd fd。这个过程经历了多次无谓的遍历。

它没有最大连接数的限制,原因是它是基于链表来存储的,但是同样有一个缺点:

1、大量的 f d fd fd的数组被整体复制于用户态和内核地址空间之间,而不管这样的复制是不是有意义。

2、 p o l l poll poll还有一个特点是“水平触发”,如果报告了 f d fd fd后,没有被处理,那么下次 p o l l poll poll时会再次报告该 f d fd fd

(3) e p o l l epoll epoll:

e p o l l epoll epoll E P O L L L T EPOLLLT EPOLLLT E P O L L E T EPOLLET EPOLLET两种触发模式, L T LT LT是默认的模式, E T ET ET是“高速”模式。 L T LT LT模式下,只要这个 f d fd fd还有数据可读,每次 e p o l l epoll epoll_ w a i t wait wait都会返回它的事件,提醒用户程序去操作,而在 E T ET ET(边缘触发)模式中,它只会提示一次,直到下次再有数据流入之前都不会再提示了,无 论 f d fd fd中是否还有数据可读。所以在 E T ET ET模式下, r e a d read read一个 f d fd fd的时候一定要把它的 b u f f e r buffer buffer读光,也就是说一直读到 r e a d read read的返回值小于请求值,或者 遇到 E A G A I N EAGAIN EAGAIN错误。还有一个特点是, e p o l l epoll epoll使用“事件”的就绪通知方式,通过 e p o l l epoll epoll_ c t l ctl ctl注册 f d fd fd,一旦该 f d fd fd就绪,内核就会采用类似 c a l l b a c k callback callback的回调机制来激活该 f d fd fd e p o l l epoll epoll_ w a i t wait wait便可以收到通知。

1.、 e p o l l epoll epoll为什么要有 E P O L L E T EPOLLET EPOLLET触发模式?

如果采用 E P O L L L T EPOLLLT EPOLLLT模式的话,系统中一旦有大量你不需要读写的就绪文件描述符,它们每次调用 e p o l l epoll epoll_ w a i t wait wait都会返回,这样会大大降低处理程序检索自己关心的就绪文件描述符的效率.。而采用 E P O L L E T EPOLLET EPOLLET这种边沿触发模式的话,当被监控的文件描述符上有可读写事件发生时, e p o l l epoll epoll_ w a i t ( ) wait() wait()会通知处理程序去读写。如果这次没有把数据全部读写完(如读写缓冲区太小),那么下次调用 e p o l l epoll epoll_ w a i t ( ) wait() wait()时,它不会通知你,也就是它只会通知你一次,直到该文件描述符上出现第二次可读写事件才会通知你!!!这种模式比水平触发效率高,系统不会充斥大量你不关心的就绪文件描述符

e p o l l epoll epoll的优点:

1、没有最大并发连接的限制,能打开的 F D FD FD的上限远大于1024(1 G G G的内存上能监听约10万个端口);
2、效率提升,不是轮询的方式,不会随着 F D FD FD数目的增加效率下降。只有活跃可用的 F D FD FD才会调用 c a l l b a c k callback callback函数;
E p o l l Epoll Epoll最大的优点就在于它只管你“活跃”的连接,而跟连接总数无关,因此在实际的网络环境中, E p o l l Epoll Epoll的效率就会远远高于 s e l e c t select select p o l l poll poll

3、 内存拷贝,利用 m m a p ( ) mmap() mmap()文件映射内存加速与内核空间的消息传递;即 e p o l l epoll epoll使用 m m a p mmap mmap减少复制开销。

s e l e c t 、 p o l l 、 e p o l l select、poll、epoll selectpollepoll 区别总结:

1、支持一个进程所能打开的最大连接数

s e l e c t select select

单个进程所能打开的最大连接数有 F D FD FD_ S E T S I Z E SETSIZE SETSIZE宏定义,其大小是32个整数的大小(在32位的机器上,大小就是3232,同理64位机器上 F D FD FD_ S E T S I Z E SETSIZE SETSIZE为3264),当然我们可以对进行修改,然后重新编译内核,但是性能可能会受到影响,这需要进一步的测试。

p o l l poll poll

p o l l poll poll本质上和 s e l e c t select select没有区别,但是它没有最大连接数的限制,原因是它是基于链表来存储的

e p o l l epoll epoll

虽然连接数有上限,但是很大,1 G G G内存的机器上可以打开10万左右的连接,2 G G G内存的机器可以打开20万左右的连接

2、 F D FD FD剧增后带来的 I O IO IO效率问题

s e l e c t select select

因为每次调用时都会对连接进行线性遍历,所以随着 F D FD FD的增加会造成遍历速度慢的“线性下降性能问题”。

p o l l poll poll

同上

e p o l l epoll epoll

因为 e p o l l epoll epoll内核中实现是根据每个 f d fd fd上的 c a l l b a c k callback callback函数来实现的,只有活跃的 s o c k e t socket socket才会主动调用 c a l l b a c k callback callback,所以在活跃 s o c k e t socket socket较少的情况下,使用 e p o l l epoll epoll没有前面两者的线性下降的性能问题,但是所有 s o c k e t socket socket都很活跃的情况下,可能会有性能问题。

3、 消息传递方式

s e l e c t select select

内核需要将消息传递到用户空间,都需要内核拷贝动作

p o l l poll poll

同上

e p o l l epoll epoll

e p o l l epoll epoll通过内核和用户空间共享一块内存来实现的。

总结:

综上,在选择 s e l e c t select select p o l l poll poll e p o l l epoll epoll时要根据具体的使用场合以及这三种方式的自身特点。

1、表面上看 e p o l l epoll epoll的性能最好,但是在连接数少并且连接都十分活跃的情况下, s e l e c t select select p o l l poll poll的性能可能比 e p o l l epoll epoll好,毕竟 e p o l l epoll epoll的通知机制需要很多函数回调。

2、 s e l e c t select select低效是因为每次它都需要轮询。但低效也是相对的,视情况而定,也可通过良好的设计改善

关于这三种 I O IO IO多路复用的用法,前面三篇总结写的很清楚,并用服务器回射 e c h o echo echo程序进行了测试。链接如下所示:

s e l e c t select selectIO多路复用之select总结

p o l l poll pollIO多路复用之poll总结

e p o l l epoll epollIO多路复用之epoll总结

今天对这三种 I O IO IO多路复用进行对比,参考网上和书上面的资料,整理如下:

1、 s e l e c t select select实现

s e l e c t select select的调用过程如下所示:

(1)使用 c o p y copy copy_ f r o m from from_ u s e r user user从用户空间拷贝 f d fd fd_ s e t set set到内核空间

(2)注册回调函数__ p o l l w a i t pollwait pollwait

(3)遍历所有 f d fd fd,调用其对应的 p o l l poll poll方法(对于 s o c k e t socket socket,这个 p o l l poll poll方法是 s o c k sock sock_ p o l l poll poll s o c k sock sock_ p o l l poll poll根据情况会调用到 t c p tcp tcp_ p o l l poll poll, u d p udp udp_ p o l l poll poll或者 d a t a g r a m datagram datagram_ p o l l poll poll

(4)以 t c p tcp tcp_ p o l l poll poll为例,其核心实现就是__ p o l l w a i t pollwait pollwait,也就是上面注册的回调函数。

(5)__ p o l l w a i t pollwait pollwait的主要工作就是把 c u r r e n t current current(当前进程)挂到设备的等待队列中,不同的设备有不同的等待队列,对于 t c p tcp tcp_ p o l l poll poll来说,其等待队列是 s k − > s k sk->sk sk>sk_ s l e e p sleep sleep(注意把进程挂到等待队列中并不代表进程已经睡眠了)。在设备收到一条消息(网络设备)或填写完文件数据(磁盘设备)后,会唤醒设备等待队列上睡眠的进程,这时 c u r r e n t current current便被唤醒了。

(6) p o l l poll poll方法返回时会返回一个描述读写操作是否就绪的 m a s k mask mask掩码,根据这个 m a s k mask mask掩码给 f d fd fd_ s e t set set赋值。

(7)如果遍历完所有的 f d fd fd,还没有返回一个可读写的 m a s k mask mask掩码,则会调用 s c h e d u l e schedule schedule_ t i m e o u t timeout timeout是调用 s e l e c t select select的进程(也就是 c u r r e n t current current)进入睡眠。当设备驱动发生自身资源可读写后,会唤醒其等待队列上睡眠的进程。如果超过一定的超时时间( s c h e d u l e schedule schedule_ t i m e o u t timeout timeout指定),还是没人唤醒,则调用 s e l e c t select select的进程会重新被唤醒获得 C P U CPU CPU,进而重新遍历 f d fd fd,判断有没有就绪的 f d fd fd

(8)把 f d fd fd_ s e t set set从内核空间拷贝到用户空间。

总结:

s e l e c t select select的几大缺点:

(1)每次调用 s e l e c t select select,都需要把 f d fd fd集合从用户态拷贝到内核态,这个开销在 f d fd fd很多时会很大

(2)同时每次调用 s e l e c t select select都需要在内核遍历传递进来的所有 f d fd fd,这个开销在 f d fd fd很多时也很大

(3) s e l e c t select select支持的文件描述符数量太小了,默认是1024

2 p o l l poll poll实现

p o l l poll poll的实现和 s e l e c t select select非常相似,只是描述 f d fd fd集合的方式不同, p o l l poll poll使用 p o l l f d pollfd pollfd结构而不是 s e l e c t select select f d fd fd_ s e t set set结构,其他的都差不多,管理多个描述符也是进行轮询,根据描述符的状态进行处理,但是 p o l l poll poll没有最大文件描述符数量的限制。 p o l l poll poll s e l e c t select select同样存在一个缺点就是,包含大量文件描述符的数组被整体复制于用户态和内核的地址空间之间,而不论这些文件描述符是否就绪,它的开销随着文件描述符数量的增加而线性增大。

3、 e p o l l epoll epoll

e p o l l epoll epoll既然是对 s e l e c t select select p o l l poll poll的改进,就应该能避免上述的三个缺点。那 e p o l l epoll epoll都是怎么解决的呢?在此之前,我们先看一下 e p o l l epoll epoll s e l e c t select select p o l l poll poll的调用接口上的不同, s e l e c t select select p o l l poll poll都只提供了一个函数—— s e l e c t select select或者 p o l l poll poll函数。而 e p o l l epoll epoll提供了三个函数, e p o l l epoll epoll_ c r e a t e create create, e p o l l epoll epoll_ c t l ctl ctl e p o l l epoll epoll_ w a i t wait wait e p o l l epoll epoll_ c r e a t e create create是创建一个 e p o l l epoll epoll句柄; e p o l l epoll epoll_ c t l ctl ctl是注册要监听的事件类型; e p o l l epoll epoll_ w a i t wait wait则是等待事件的产生。

对于第一个缺点, e p o l l epoll epoll的解决方案在 e p o l l epoll epoll_ c t l ctl ctl函数中。每次注册新的事件到 e p o l l epoll epoll句柄中时(在 e p o l l epoll epoll_ c t l ctl ctl中指定 E P O L L EPOLL EPOLL_ C T L CTL CTL_ A D D ADD ADD),会把所有的 f d fd fd拷贝进内核,而不是在 e p o l l epoll epoll_ w a i t wait wait的时候重复拷贝。 e p o l l epoll epoll保证了每个 f d fd fd在整个过程中只会拷贝一次。

对于第二个缺点, e p o l l epoll epoll的解决方案不像 s e l e c t select select p o l l poll poll一样每次都把 c u r r e n t current current轮流加入 f d fd fd对应的设备等待队列中,而只在 e p o l l epoll epoll_ c t l ctl ctl时把 c u r r e n t current current挂一遍(这一遍必不可少)并为每个 f d fd fd指定一个回调函数,当设备就绪,唤醒等待队列上的等待者时,就会调用这个回调函数,而这个回调函数会把就绪的 f d fd fd加入一个就绪链表)。 e p o l l epoll epoll_ w a i t wait wait的工作实际上就是在这个就绪链表中查看有没有就绪的 f d fd fd(利用 s c h e d u l e schedule schedule_ t i m e o u t ( ) timeout() timeout()实现睡一会,判断一会的效果,和 s e l e c t select select实现中的第7步是类似的)。

对于第三个缺点, e p o l l epoll epoll没有这个限制,它所支持的 F D FD FD上限是最大可以打开文件的数目,这个数字一般远大于2048,举个例子,在1 G B GB GB内存的机器上大约是10万左右,具体数目可以 c a t cat cat / p r o c / s y s / f s / f i l e − m a x /proc/sys/fs/file-max /proc/sys/fs/filemax察看,一般来说这个数目和系统内存关系很大。

总结:

(1) s e l e c t select select p o l l poll poll实现需要自己不断轮询所有 f d fd fd集合,直到设备就绪,期间可能要睡眠和唤醒多次交替。而 e p o l l epoll epoll其实也需要调用 e p o l l epoll epoll w a i t wait wait不断轮询就绪链表,期间也可能多次睡眠和唤醒交替,但是它是设备就绪时,调用回调函数,把就绪 f d fd fd放入就绪链表中,并唤醒在 e p o l l epoll epoll w a i t wait wait中进入睡眠的进程。虽然都要睡眠和交替,但是 s e l e c t select select p o l l poll poll在“醒着”的时候要遍历整个 f d fd fd集合,而 e p o l l epoll epoll在“醒着”的时候只要判断一下就绪链表是否为空就行了,这节省了大量的 C P U CPU CPU时间。这就是回调机制带来的性能提升。

(2) s e l e c t select select p o l l poll poll每次调用都要把 f d fd fd集合从用户态往内核态拷贝一次,并且要把 c u r r e n t current current往设备等待队列中挂一次,而 e p o l l epoll epoll只要一次拷贝,而且把 c u r r e n t current current往等待队列上挂也只挂一次(在 e p o l l epoll epoll_ w a i t wait wait的开始,注意这里的等待队列并不是设备等待队列,只是一个 e p o l l epoll epoll内部定义的等待队列)。这也能节省不少的开销。

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

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值