select模块以及IO多路复用
前言
Python中的select
模块专注于I/O多路复用,提供了select
, poll
, epoll
三个方法(其中后两个在Linux中可用,windows仅支持select),另外也提供了kqueue
方法(freeBSD系统).
select()的机制中提供一fd_set
的数据结构,实际上是一long
类型的数组, 每一个数组元素都能与一打开的文件句柄(不管是Socket句柄,还是其他文件或命名管道或设备句柄)建立联系,建立联系的工作由程序员完成, 当调用select()时,由内核根据IO状态修改fd_set的内容,由此来通知执行了select()的进程哪一Socket或文件可读或可写。
select主要用于socket通信当中,能监视我们需要的文件描述符变化。
非阻塞式I/O编程特点
- 如果发现一个I/O有输入,读取的过程中,另外一个也有了输入,这时候不会产生任何反应.这就需要你的程序语句去用到select函数的时候才知道有数据输入。
- 程序去select的时候,如果没有数据输入,程序会一直等待,直到有数据为止,也就是程序中无需循环和sleep。
Select在Socket编程中还是比较重要的,可是对于初学Socket的人来说都不太爱用Select写程序,他们只是习惯写诸如connect
、accept
、recv
或recvfrom
这样的阻塞程序(所谓阻塞方式block,顾名思义,就是进程或是线程执行到这些函数时必须等待某个事件的发生,如果事件没有发生,进程或线程就被阻塞,函数不能立即返回)。
可是使用Select就可以完成非阻塞(所谓非阻塞方式non-block,就是进程或线程执行此函数时不必非要等待事件的发生,一旦执行肯定返回,以返回值的不同来反映函数的执行情况,如果事件发生则与阻塞方式相同,若事件没有发生,则返回一个代码来告知事件未发生,而进程或线程继续执行,所以效率较高)方式工作的程序,它能够监视我们需要监视的文件描述符的变化情况——读写或是异常。
Select方法
基本原理
进程指定内核监听哪些文件描述符(最多监听1024个fd)的哪些事件,当没有文件描述符事件发生时,进程被阻塞;当一个或者多个文件描述符事件发生时,进程被唤醒。
当我们调用select()时:
- 上下文切换转换为内核态
- 将fd从用户空间复制到内核空间
- 内核遍历所有fd,查看其对应事件是否发生
- 如果没发生,将进程阻塞,当设备驱动产生中断或者timeout时间后,将进程唤醒,再次进行遍历
- 返回遍历后的fd
- 将fd从内核空间复制到用户空间
select函数方法参数
1 | fd_r_list, fd_w_list, fd_e_list = select.select(rlist, wlist, xlist, [timeout]) |
参数
可接受四个参数(前三个必须):
- rlist: wait until ready for reading
- wlist: wait until ready for writing
- xlist: wait for an “exceptional condition”
- timeout: 超时时间
返回值:三个列表
select方法用来监视文件描述符(当文件描述符条件不满足时,select会阻塞),当某个文件描述符状态改变后,会返回三个列表
- 当参数1 序列中的fd满足“可读”条件时,则获取发生变化的fd并添加到fd_r_list中
- 当参数2 序列中含有fd时,则将该序列中所有的fd添加到 fd_w_list中
- 当参数3 序列中的fd发生错误时,则将该发生错误的fd添加到 fd_e_list中
- 当超时时间为空,则select会一直阻塞,直到监听的句柄发生变化.当超时时间 = n(正整数)时,那么如果监听的句柄均无任何变化,则select会阻塞n秒,之后返回三个空列表,如果监听的句柄有变化,则直接执行。
示例
示例1:模拟select,同时监听多个端口
- 服务端
1 | # coding=utf-8 |
- 客户端
1 | # coding=utf-8 |
1 | # coding=utf-8 |
运行server端和client端,客户端1,2均能连接。
但是以上程序并不能同时对客户端的输入同时响应处理(两个客户端连接都没关闭的情况下),下面就来介绍I/O多路复用的例子
示例2:IO多路复用–使用socket模拟多线程,并实现读写分离
- 服务端
1 | # coding=utf-8 |
- 客户端
1 | # coidng=utf-8 |
- 运行结果
分别运行服务端和客户端程序:
1 | python select_multi_server.py |
1 | python select_multi_client.py |
多次运行程序,你会发现客户端程序返回结果里的received后面的略有不同,你发现其中的原因了吗!
select、poll、epoll区别
select
, poll
, epoll
都是I/O多路复用的具体的实现,之所以有这三个存在,其实是因为他们出现是有先后顺序的。
I/O多路复用这个概念被提出来以后, select是第一个实现 (1983 左右在BSD里面实现的)。
select
select 被实现以后,很快就暴露出了很多问题:
- select 会修改传入的参数数组,这个对于一个需要调用很多次的函数,是非常不友好的。
- 每次调用select,都需要把fd集合从用户态拷贝到内核态,这个开销在fd很多时会很大
- select 如果任何一个sock(I/O stream)出现了数据,select仅仅会返回,但是并不会告诉你是那个sock上有数据,于是你只能自己一个一个的找,)每次调用select都需要在内核遍历传递进来的所有fd,这个开销在fd很多时也很大
- select 只能监视1024个链接,linux 定义在头文件中的,参见
FD_SETSIZE
。 - select 不是线程安全的,如果你把一个sock加入到select, 然后突然另外一个线程发现,尼玛,这个sock不用,要收回。对不起,这个select 不支持的,如果你丧心病狂的竟然关掉这个sock, select的标准行为是。。呃。。不可预测的,
于是14年以后(1997年)一帮人又实现了poll
, poll
修复了select
的很多问题
poll
- poll 去掉了1024个链接的限制,于是要多少链接呢, 主人你开心就好。
- poll 从设计上来说,不再修改传入数组,不过这个要看你的平台了,所以行走江湖,还是小心为妙。
其实拖14年那么久也不是效率问题, 而是那个时代的硬件实在太弱,一台服务器处理1千多个链接简直就是神一样的存在了,select
很长段时间已经满足需求。
但是poll
仍然不是线程安全的, 这就意味着,不管服务器有多强悍,你也只能在一个线程里面处理一组I/O流。你当然可以那多进程来配合了,不过然后你就有了多进程的各种问题。
于是5年以后, 在2002, 大神 Davide Libenzi 实现了epoll
.
epoll
epoll
可以说是I/O 多路复用最新的一个实现,epoll
修复了poll
和select
绝大部分问题, 比如:
- 对于每次需要将FD从用户态拷贝至内核态.epoll的解决方案在epoll_ctl函数中。每次注册新的事件到epoll句柄中时(在epoll_ctl中指定EPOLL_CTL_ADD),会把所有的fd拷贝进内核,而不是在epoll_wait的时候重复拷贝。epoll保证了每个fd在整个过程中只会拷贝一次。
- 同样epoll也没有1024的连接数限制
- epoll 现在是线程安全的。
- epoll 现在不仅告诉你sock组里面数据,还会告诉你具体哪个sock有数据,你不用自己去找了。
epoll
的解决方案不像select
或poll
一样每次都把current轮流加入fd对应的设备等待队列中,而只在epoll_ctl
时把current挂一遍(这一遍必不可少)并为每个fd指定一个回调函数,当设备就绪,唤醒等待队列上的等待者时,就会调用这个回调函数,而这个回调函数会把就绪的fd加入一个就绪链表)。epoll_wait
的工作实际上就是在这个就绪链表中查看有没有就绪的fd(利用schedule_timeout()
实现睡一会,判断一会的效果,和select实现中的第7步是类似的)。
select/poll, epoll总结
select
,poll
实现需要自己不断轮询所有fd集合,直到设备就绪,期间可能要睡眠和唤醒多次交替。而epoll其实也需要调用epoll_wait不断轮询就绪链表,期间也可能多次睡眠和唤醒交替,但是它是设备就绪时,调用回调函数,把就绪fd放入就绪链表中,并唤醒在epoll_wait中进入睡眠的进程。虽然都要睡眠和交替,但是select和poll在“醒着”的时候要遍历整个fd集合,而epoll在“醒着”的时候只要判断一下就绪链表是否为空就行了,这节省了大量的CPU时间。这就是回调机制带来的性能提升。select,poll每次调用都要把fd集合从用户态往内核态拷贝一次,并且要把current往设备等待队列中挂一次,而epoll只要一次拷贝,而且把current往等待队列上挂也只挂一次(在epoll_wait的开始,注意这里的等待队列并不是设备等待队列,只是一个epoll内部定义的等待队列)。这也能节省不少的开销。
epoll示例1:简单时间戳服务器
- 服务端
1 | import socket |
- 客户端
1 | import socket |
epoll示例2:读写分离的epoll
- 服务端
1 | #!/usr/bin/env python |
- 客户端
1 | import socket |
小结
本文总结了I/O多路复用的三种方式select、poll、epoll,并使用python下select模块实现了以其为基础的时间戳服务端和客户端。