python 并发编程 多路复用IO模型详解


Posted in Python onAugust 20, 2019

多路复用IO(IO multiplexing)

这种IO方式为事件驱动IO(event driven IO)。

我们都知道,select/epoll的好处就在于单个进程process就可以同时处理多个网络连接的IO。它的基本原理就是select/epoll这个function会不断的轮询所负责的所有socket,当某个socket有数据到达了,就通知用户进程。它的流程如图:

python 并发编程 多路复用IO模型详解

select是多路复用的一种

当用户进程调用了select,那么整个进程会被block,而同时,kernel会“监视”所有select负责的socket,
当任何一个socket中的数据准备好了,select就会返回。这个时候用户进程再调用read操作,将数据从kernel拷贝到用户进程。
这个图和blocking IO的图其实并没有太大的不同,事实上还更差一些。因为这里需要使用两个系统调用\(select和recvfrom\),
而blocking IO只调用了一个系统调用\(recvfrom\)。但是,用select的优势在于它可以同时处理多个connection。

多路复用IO比较阻塞IO模型:

1.阻塞IO经历两个阶段 wait data,copy data

2.多路复用3个阶段 wait data,ready copy data, copy data

单连接套接字通信 阻塞IO效率高

多路复用IO select可以代理多个套接字连接,多个套接字通信,多路复用IO效率高

强调:

1. 如果处理的连接数不是很高的话,使用select/epoll的web server不一定比使用multi-threading + blocking IO的web server性能更好,可能延迟还更大。select/epoll的优势并不是对于单个连接能处理得更快,而是在于能处理更多的连接。

2. 在多路复用模型中,对于每一个socket,一般都设置成为non-blocking,但是,如上图所示,整个用户的process其实是一直被block的。只不过process是被select这个函数block,而不是被socket IO给block。

结论: select的优势在于可以处理多个连接,性能高,同时可以检测多个套接字IO行为,不适用于单个连接

select网络IO模型示例

select 检测多个套接字IO行为 accept,recv

IO行为两种:

1.别人给我传数据

2.给别人发送数据

timeout是超时时间

每隔0.5秒去问操作系统准备好数据没有

def select(rlist, wlist, xlist, timeout=None): 
  pass
# [] 传的空列表是出异常的列表
# 返回值3个列表 收列表,发列表,异常列表
rl,wl,xl = select.select(rlist, wlist, [], 0.5)

客户端:

from socket import *
client = socket(AF_INET,SOCK_STREAM)
client.connect(('127.0.0.1',8000))
while True:
  msg = input(">>>:").strip()
  if not msg:continue
  client.send(msg.encode("utf-8"))
  data = client.recv(1024)
  print(data.decode("utf-8"))
client.close()

服务端代码:

from socket import *
import select
server = socket(AF_INET,SOCK_STREAM)
server.bind(('127.0.0.1',8000))
server.listen(5)
# 设置socket接口为 非阻塞IO接口
# 默认是True 为阻塞
server.setblocking(False)
# 专门存着收消息套接字
rlist = [server,]
# 存放发送消息套接字
wlist = []
# 存放发送的数据
wdata = {}
while True:
  # 返回值3个列表 收列表,发列表,异常列表
  rl,wl,xl = select.select(rlist, wlist, [], 0.5)
  print("rl",rl)
  print("wl",wl)
  for sock in rl:
    if sock == server:
      conn,addr = sock.accept()
      rlist.append(conn)
    else:
      try:
        data = sock.recv(1024)
        if not data:
          sock.close()
          rlist.remove(sock)
          continue
        # 收的套接字加到列表
        wlist.append(sock)
        # 把数据加到字典 做一个 套接字对应数据
        wdata[sock] = data.upper()

      except Exception:
        sock.close()
        rlist.remove(sock)
  # 发送数据
  for sock in wl:
    sock.send(wdata[sock])
    wlist.remove(sock)
    wdata.pop(sock)
server.close()

基于select模块 检测套接字IO行为,实现并发效果

select监听fd变化的过程分析:

用户进程创建socket对象,拷贝监听的fd到内核空间,每一个fd会对应一张系统文件表,内核空间的fd响应到数据后,
就会发送信号给用户进程数据已到;

用户进程再发送系统调用,比如(accept)将内核空间的数据copy到用户空间,同时作为接受数据端内核空间的数据清除,
这样重新监听时fd再有新的数据又可以响应到了(发送端因为基于TCP协议所以需要收到应答后才会清除)。

该模型的优点:

可以同时检测多个套接字,效率比阻塞IO,非阻塞IO高了

相比其他模型,使用select() 的事件驱动模型只用单线程(进程)执行,占用资源少,不消耗太多 CPU,同时能够为多客户端提供服务。

如果试图建立一个简单的事件驱动的服务器程序,这个模型有一定的参考价值。

该模型的缺点:

代理的套接字 列表里的多个套接字,需要循环列表 一个个检测,

在代理套接字比较少的情况下,循环比较快。但select代理的套接字非常多的情况下,select随着列表增大,效率就越来越慢

首先select()接口并不是实现“事件驱动”的最好选择。因为当需要探测的句柄值较大时,select()接口本身需要消耗大量时间去轮询各个句柄。

很多操作系统提供了更为高效的接口,如linux提供了epoll,BSD提供了kqueue,Solaris提供了/dev/poll,…。
如果需要实现更高效的服务器程序,类似epoll这样的接口更被推荐。遗憾的是不同的操作系统特供的epoll接口有很大差异,
所以使用类似于epoll的接口实现具有较好跨平台能力的服务器会比较困难。
其次,该模型将事件探测和事件响应夹杂在一起,一旦事件响应的执行体庞大,则对整个模型是灾难性的。

epoll是异步方式实现,提交套接字时候,每个套接字身上都绑定一个回调函数,哪个套接字准备好了,就触发回调函数,把自己索引放在单独列表里,对于select来说,只需要去准备好的列表里 根据索引拿到套接字,这样不需要在列表里每个遍历。

epoll不支持windows系统

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
Python编程使用NLTK进行自然语言处理详解
Nov 16 Python
使用实现pandas读取csv文件指定的前几行
Apr 20 Python
python3.6实现学生信息管理系统
Feb 21 Python
python中报错"json.decoder.JSONDecodeError: Expecting value:"的解决
Apr 29 Python
python语言基本语句用法总结
Jun 11 Python
python算法与数据结构之单链表的实现代码
Jun 27 Python
python3实现斐波那契数列(4种方法)
Jul 15 Python
浅谈Python3 numpy.ptp()最大值与最小值的差
Aug 24 Python
在django admin详情表单显示中添加自定义控件的实现
Mar 11 Python
Python Django中的STATIC_URL 设置和使用方式
Mar 27 Python
Python while true实现爬虫定时任务
Jun 08 Python
Python 读取位于包中的数据文件
Aug 07 Python
关于pytorch中网络loss传播和参数更新的理解
Aug 20 #Python
对pytorch中的梯度更新方法详解
Aug 20 #Python
PyTorch: 梯度下降及反向传播的实例详解
Aug 20 #Python
python爬虫 urllib模块发起post请求过程解析
Aug 20 #Python
pytorch 加载(.pth)格式的模型实例
Aug 20 #Python
python multiprocessing模块用法及原理介绍
Aug 20 #Python
python 并发编程 阻塞IO模型原理解析
Aug 20 #Python
You might like
PHP如何解决网站大流量与高并发的问题
2011/06/25 PHP
php XPath对XML文件查找及修改实现代码
2011/07/27 PHP
php实现简易聊天室应用代码
2015/09/23 PHP
PHP编写daemon process 实例详解
2016/11/13 PHP
原生js拖拽(第一课 未兼容)拖拽思路
2013/03/29 Javascript
JS实现切换标签页效果实例代码
2013/11/01 Javascript
鼠标左键单击冲突的问题解决方法(防止冒泡)
2014/05/14 Javascript
完美解决jQuery fancybox ie 无法显示关闭按钮的问题
2016/11/29 Javascript
js实现打地鼠小游戏
2017/02/13 Javascript
jquery实现的table排序功能示例
2017/03/10 Javascript
javascript中this用法实例详解
2017/04/06 Javascript
JavaScript在控件上添加倒计时功能的实现代码
2017/07/04 Javascript
Async/Await替代Promise的6个理由
2019/06/15 Javascript
微信小程序与公众号实现数据互通的方法
2019/07/25 Javascript
p5.js临摹动态图形实现方法详解
2019/10/23 Javascript
JavaScript实现联动菜单特效
2020/01/07 Javascript
python修改操作系统时间的方法
2015/05/18 Python
Python中urllib+urllib2+cookielib模块编写爬虫实战
2016/01/20 Python
使用Python绘制图表大全总结
2017/02/11 Python
Python/ArcPy遍历指定目录中的MDB文件方法
2018/10/27 Python
澳大利亚家具和家居用品在线商店:Interiors Online
2018/03/05 全球购物
日本高岛屋百货购物网站:TAKASHIMAYA
2019/03/24 全球购物
小米乌克兰网上商店:Xiaomi.UA
2019/10/29 全球购物
JSF界面控制层技术
2013/06/17 面试题
在校生钳工实习自我鉴定
2013/09/19 职场文书
优秀党员获奖感言
2014/02/18 职场文书
党风廉政建设责任书
2014/04/14 职场文书
化学教育专业求职信
2014/07/08 职场文书
工厂门卫的岗位职责
2014/07/27 职场文书
银行催款通知书
2015/04/17 职场文书
冬季作息时间调整通知
2015/04/24 职场文书
信访维稳承诺书
2015/05/04 职场文书
民间借贷被告代理词
2015/05/23 职场文书
刘胡兰观后感
2015/06/16 职场文书
无房证明样本
2015/06/17 职场文书
积极心理学课程心得体会
2016/01/22 职场文书