select,poll,epoll区别总结
select,poll,epoll都是I/O多路复用。I/O多路复用就是通过一种机制,可以监测多个描述符,一旦某个描述就绪(一般是读或者写),能够通知程序进行相应的读写操作。但select、poll、epoll本质上都是同步I/O,因为他们都需要在读写事件就绪后自己负责进行读写,也就是说这个读写过程是阻塞的,而异步I/O则无需自己负责进行读写
,异步I/O的实现会负责把数据从内核拷贝到用户空间。
I/O多路复用之select总结
I/O多路复用是指内核一旦发现进程指定的一个或者多个I/O条件准备读取,它就通知进程。适用于以下场合:
1、当客户处理多个描述字时(一般是交互式输入和网络套接口),必须使用I/O复用。
2、当一个客户同时处理多个套接口时,而这种情况是可能的,但很少出现。
3、如果一个TCP服务器既要处理监听套接口,又要处理已连接套接口,一般也要用到I/O复用。
4、如果一个服务器即要处理TCP,又要处理UDP,一般要使用I/O复用。
5、如果一个服务器要处理多个服务或多个协议,一般要使用I/O复用。
6、与多进程和多线程技术相比,I/O多路复用技术的最大优势是系统开销小,系统不必创建进程/线程维护这些进程/线程,从而大大减少了系统的开销。
select的调用过程
1、使用copy_from_user从用户空间拷贝fd_set到内核空间
2、注册回调函数__pollwait
3、遍历所有fd,调用其对应的poll方法(对于socket,这个poll方法是sock_poll,sock_poll根据情况会调用到tcp_poll,udp_poll或者datagram_poll)
4、以tcp_poll为例,其核心实现就是__pollwait,也就是上面注册的回调函数。
5、__pollwait的主要工作就是把current(当前进程)挂到设备的等待队列中,不同的设备有不同的等待队列,对于tcp_poll来说,其等待队列是sk->sk_sleep(注意把进程挂到等待队列中并不代表进程已经睡眠了)。在设备收到一条消息(网络设备)或填写完文件数据(磁盘设备)后,会唤醒设备等待队列上睡眠的进程,这时current便被唤醒了。
6、poll方法返回时会返回一个描述读写操作是否就绪的mask掩码,根据这个mask掩码给fd_set赋值。
7、如果遍历完所有的fd,还没有返回一个可读写的mask掩码,则会调用schedule_timeout是调用select的进程(也就是current)进入睡眠。当设备驱动发生自身资源可读写后,会唤醒其等待队列上睡眠的进程。如果超过一定的超时时间(schedule_timeout指定),还是没人唤醒,则调用select的进程会重新被唤醒获得CPU,进而重新遍历fd,判断有没有就绪的fd。
8、把fd_set从内核空间拷贝到用户空间。
select的几大缺点:
1、每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时会很大。
2、同时每次调用select都需要在内核遍历传递进来的fd,这个开销在fd很多时也很大
3、select支持的文件描述符数量太小了,默认是1024
I/O多路复用之poll总结
poll的实现和select非常相似,只是描述fd集合的方式不同,poll使用的是pollfd结构而不是select的fd_set结构,其他的都差不多,管理多个描述符也是进行轮询,根据描述符的状态进行处理,但poll没有最大文件描述符数量的限制。poll和select同样存在一个缺点就是,包含大量文件描述符的数组被整体复制于用户态和内核态的地址空间,而不论这些文件描述符是否就绪,它的开销随着文件描述符数量的增加而线性增大。
I/O多路复用之epoll总结
1、mmap
mmap系统调用并不是为了共享内存而设计的,它提供了不同于一般对普通文件的访问方式,进程可以像都内存一样对普通文件操作。
mmap系统调用使得进程之间通过映射同一个普通的文件实现共享。普通文件被映射到进程地址空间后,进程可以像访问普通内存一样对文件进行访问,不必再调用read()/write()等操作
mmap并不分配空间,只是将文件映射到嗲用进程的地址空间(占用虚拟地址空间),然后就可以使用memcpy()等操作,内存中文件并不立即更新到文件中,而是由一段时间的延迟,可以使用msync()显式的同步。
优点:
1.1、提升效率
一般写文件需要用open 、read、write,需要将磁盘文件读取到内核的cache缓冲区,然后再拷贝到用户内空间内存,涉及两次的读写。
mmap通过将磁盘文件映射到用户空间,当进程读文件时,发生缺页中断,给虚拟内存分配物理内存,在通过磁盘调页操作将磁盘数据读到物理内存上面,实现了用户空间数据的读取,整个过程只有一次内存的拷贝。
1.2、用于进程间大量的数据通信
两个进程映射同一个文件,在两个进程中,同一个文件去映射的虚拟地址空间不同。一个进程操作文件时,先通过缺页获取物理地址。进而通过磁盘文件调页操作将文件读入到内存。
另一个进程访问文件的时候,发现没有物理页面映射到虚拟内存,通过FS的缺页处理查找cache区是否有读入磁盘文件,有的话建立映射关系,这两个进程通过共享内存就可以进行通讯了。
mmap把文件映射到内存空间中,简单的说mmap就是把一个文件的内容做一个映射。映成功后,用户对这段内存区域的修改就可以直接反应到内核空间,同样内核空间对这段区域的修改也可以直接反应到用户的空间。那么对与内核空间------用户空间两者之间需要大量数据传输等操作的话效率是非常高的。
mmap并不分配内存空间,只是将文件映射到调用进程的地址空间里(但是会占用你的virutal memory),然后你就可以用memery等操作写文件,而不用write()了。写完后,内存中的文件不会立即更新到文件中,而是有一段时间的延迟,可以使用msync()来显式同步一下,这样你所写的内容就能立马保存到文件里面了。不过通过mmap来写文件不能增加文件的长度,应为映射的长度在调用mmap()的时候就决定了。
原理
首先,映射,就是建立一种对应的关系,主要是在硬盘上文件的位置与进程地址空间中一块大小相同的区域之间--对应。
在内存映射的过程中,并没有实际数据的拷贝,文件并没与被载入到内存中,只是逻辑上被放入到了内存,具体到代码,就是你建立并初始化了相关的数据结构。
实际上,进程之间在共享内存时,并不总是读写少量数据后就解除映射,有新进程通讯,在重新建立共享内存区域。而是保持共享区域,知道通信完毕为止,这样,数据内容一致保存在共享内存中,并没有写回文件。共享内存中的内容往往是解除映射的时候才写回文件。因此,采用共享内存的通讯方式效率是很高的。
2、红黑树
3、链表
。。。
epoll既然是对select和poll的改进,就应该能避免上述的三个缺点。那epoll都是怎么解决的呢?在此之前,我们先看一下epoll和select和poll的调用接口上的不同,select和poll都只提供了一个函数——select或者poll函数。而epoll提供了三个函数,epoll_create,epoll_ctl和epoll_wait,epoll_create是创建一个epoll句柄;epoll_ctl是注册要监听的事件类型;epoll_wait则是等待事件的产生。
对于第一个缺点,epoll的解决方案在epoll_ctl函数中。每次注册新的事件到epoll句柄中时(在epoll_ctl中指定EPOLL_CTL_ADD),会把所有的fd拷贝进内核,而不是在epoll_wait的时候重复拷贝。epoll保证了每个fd在整个过程中只会拷贝一次。
对于第二个缺点,epoll的解决方案不像select或poll一样每次都把current轮流加入fd对应的设备等待队列中,而只在epoll_ctl时把current挂一遍(这一遍必不可少)并为每个fd指定一个回调函数,当设备就绪,唤醒等待队列上的等待者时,就会调用这个回调函数,而这个回调函数会把就绪的fd加入一个就绪链表)。epoll_wait的工作实际上就是在这个就绪链表中查看有没有就绪的fd(利用schedule_timeout()实现睡一会,判断一会的效果,和select实现中的第7步是类似的)。
对于第三个缺点,epoll没有这个限制,它所支持的FD上限是最大可以打开文件的数目,这个数字一般远大于2048,举个例子,在1GB内存的机器上大约是10万左右,具体数目可以cat /proc/sys/fs/file-max察看,一般来说这个数目和系统内存关系很大。
select、poll、epoll之间的区别总结
select、poll、epoll都是IO多路复用的机制。IO多路复用就是通过一种机制,可以监测多个描述符,一旦某个描述符就绪(一般是读就绪或者写就绪),能够通知程序进行相应的读写。select、poll、epoll本职上都是同步I/O。因为他们都需要在读事件就绪后自己负责读写,也就是说这几个读写过程是阻塞的,而异步I/O则无需自己负责读写,异步I/O的实现会负责把数据从内核拷贝到用户空间。
总结
(1)select,poll实现需要自己不断轮询所有fd集合,直到设备就绪,期间可能要睡眠和唤醒多次交替。而epoll其实也需要调用epoll_wait不断轮询就绪链表,期间也可能多次睡眠和唤醒交替,但是它是设备就绪时,调用回调函数,把就绪fd放入就绪链表中,并唤醒在epoll_wait中进入睡眠的进程。虽然都要睡眠和交替,但是select和poll在“醒着”的时候要遍历整个fd集合,而epoll在“醒着”的时候只要判断一下就绪链表是否为空就行了,这节省了大量的CPU时间。这就是回调机制带来的性能提升。
(2)select,poll每次调用都要把fd集合从用户态往内核态拷贝一次,并且要把current往设备等待队列中挂一次,而epoll只要一次拷贝,而且把current往等待队列上挂也只挂一次(在epoll_wait的开始,注意这里的等待队列并不是设备等待队列,只是一个epoll内部定义的等待队列)。这也能节省不少的开销。
1、select==>时间复杂度O(n)
它仅仅知道了,有I/O事件发生了,却并不知道是哪那几个流(可能有一个,多个,甚至全部),我们只能无差别轮询所有流,找出能读出数据,或者写入数据的流,对他们进行操作。所以select具有O(n)的无差别轮询复杂度,同时处理的流越多,无差别轮询时间就越长。
2、poll==>时间复杂度O(n)
poll本质上和select没有区别,它将用户传入的数组拷贝到内核空间,然后查询每个fd对应的设备状态, 但是它没有最大连接数的限制,原因是它是基于链表来存储的.
(3)epoll==>时间复杂度O(1)
epoll可以理解为event poll,不同于忙轮询和无差别轮询,epoll会把哪个流发生了怎样的I/O事件通知我们。所以我们说epoll实际上是事件驱动(每个事件关联上fd)的,此时我们对这些流的操作都是有意义的。(复杂度降低到了O(1))
select,poll,epoll都是IO多路复用的机制。I/O多路复用就通过一种机制,可以监视多个描述符,一旦某个描述符就绪(一般是读就绪或者写就绪),能够通知程序进行相应的读写操作。但select,poll,epoll本质上都是同步I/O,因为他们都需要在读写事件就绪后自己负责进行读写,也就是说这个读写过程是阻塞的,而异步I/O则无需自己负责进行读写,异步I/O的实现会负责把数据从内核拷贝到用户空间。
epoll跟select都能提供多路I/O复用的解决方案。在现在的Linux内核里有都能够支持,其中epoll是Linux所特有,而select则应该是POSIX所规定,一般操作系统均有实现
select
select的本质是通过设置或者检查存放fd标志位的数据结构来进行下一步处理。这样带来的缺点:
1、单个进程可监视的fd数量有限,即能监听端口号的大小有限。
一般来说这个数目和系统内存关系很大,具体数目可以cat /proc/sys/fs/file-max察看。32位机默认是1024个。64位机默认是2048.
2、对socket进行扫描是线性扫描,即采用轮训的方法,效率低:
当套接字比较多的时候,每次select()都要通过遍历FD_SETSIZE个Socket来完成调度,不管哪个Socket是活跃的,都遍历一遍。这会浪费很多CPU时间。如果能给套接字注册某个回调函数,当他们活跃时,自动完成相关操作,那就避免了轮询,这正是epoll与kqueue做的。
3、需要维护一个存放大量fd的数据结构,这样会使得用户空间和内核空间在传递该结构复制开销大。
poll
poll本质上和select没有区别,它将用户传入的数组拷贝到内核空间,然后查询每个fd对应的设备状态,如果设备就绪则在设备等待队列中加入一项并继续遍历,如果遍历完所有fd后没有发现就绪设备,则挂起当前进程,直到设备就绪或者主动超时,被唤醒后它又要再次遍历fd。这个过程经历了多次无谓的遍历。
它没有最大连接数的限制,原因是它是基于链表来存储的,但是同样有一个缺点:
1、大量的fd数组被整体复制于用户状态和内核状态之间,而不管这样的复制是不是有意义。
2、poll还有一个特点是“水平触发”,如果报告了fd后,没有被处理,那么下次poll时还会再次报告该fd。
epoll
epoll有EPOLLLT和EPOLLET两种触发模式,LT是默认的模式,ET是“高速”模式。LT模式下,只要这个fd还有数据可读,每次 epoll_wait都会返回它的事件,提醒用户程序去操作,而在ET(边缘触发)模式中,它只会提示一次,直到下次再有数据流入之前都不会再提示了,无 论fd中是否还有数据可读。所以在ET模式下,read一个fd的时候一定要把它的buffer读光,也就是说一直读到read的返回值小于请求值,或者 遇到EAGAIN错误。还有一个特点是,epoll使用“事件”的就绪通知方式,通过epoll_ctl注册fd,一旦该fd就绪,内核就会采用类似callback的回调机制来激活该fd,epoll_wait便可以收到通知。
epoll为什么要有EPOLLET模式
如果采用EPOLLLT模式的话,系统中一旦有大量你不需要读写的就绪文件描述符,它们每次调用epoll_wait都会返回,这样会大大降低处理程序检索自己关心的就绪文件描述符的效率.。而采用EPOLLET这种边沿触发模式的话,当被监控的文件描述符上有可读写事件发生时,epoll_wait()会通知处理程序去读写。如果这次没有把数据全部读写完(如读写缓冲区太小),那么下次调用epoll_wait()时,它不会通知你,也就是它只会通知你一次,直到该文件描述符上出现第二次可读写事件才会通知你!!!这种模式比水平触发效率高,系统不会充斥大量你不关心的就绪文件描述符。
epoll的优点
1、没有最大并发连接数的限制,能打开的FD的上限远大于1024(1G的内存上能监听约10万个端口)
2、效率提升,不是轮询的方式,不会随着FD数目的增加效率下降。只有活跃可用的FD才会调用callback函数;
即epoll最大的优点在于它只管你“活跃的链接”,而跟连接总数无关,因此在实际的网络环境中,Epoll的效率会远高于select和poll。
3、内存拷贝,利用mmap()文件映射内存加速与内核空间的消息传递;即内核epoll使用mmap减少复制的开销。
select、poll、epoll区别总结:
1、支持一个进程所能打开的最大连接数
1、select
单个进程所能打开的最大连接数有FD_SETSIZE宏定义,其大小是32个整数的大小(在32位的机器上,大小就是3232,同理64位机器上FD_SETSIZE为3264),当然我们可以对进行修改,然后重新编译内核,但是性能可能会受到影响,这需要进一步的测试。
2、poll
poll本质上和select没有区别,但是它没有最大连接数的限制,原因是它用链表来存储。
3、epoll
虽然连接数有上限,1G内存的机器上可以连接10万左右的连接,2G内存的机器可以打开20万左右的连接
2、fd剧增后带来的IO效率问题
1、select
因为每次调用都会对连接进行线性的遍历,所以FD的增加会造成遍历速度的“线性下降性能问题”
2、poll
同上
3、epoll
因为epoll内核中实现是根据每个fd的callback函数实现的,只有活跃的socket才会主动调用callback,所以在活跃socket较少的情况下,使用epoll没有前面两者的线性下降的性能问题,但是所有socket都很活跃的情况下,可能会有性能问题。
3、消息传递
1、select
内核需要将消息传递到用户空间,都需要内核的拷贝动作
2、poll
同上
3、epoll
通过内核和用户空间共享一块内存来实现
总结
1、表面行看epoll的想能最好,但是在连接数少并且都十分活跃的情况下,select和poll的想能可能更好,epoll的通讯机制需要很多的函数回调。
2、select低效因为每次它都需要轮训,但是低效也是相对的,视情况决定。