并发编程/IO模型

背景概念

IO模型概念

IO模型分类

阻塞IO  (blocking IO)

特点

    两个阶段(等待数据和拷贝数据两个阶段)都被block

设置

  1. server.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)

解决方案: 

    启用多线程或者多进程,要阻塞只阻塞当前线程/进程,不会影响其他进程/线程

不良影响:

    当遇到过多得链接请求时会严重占用资源,降低响应效率

修复不良影响:

    启用进程池/线程池 ,降低进程/线程数量

 仍旧未解决的不良影响:

    池得数量不好规范,请求数量和池大小不好对应,且池存在数量上限

    只是一定程度上限制了不良影响,无法根本解决

  总结 :

    基于池得创建可以解决小规模得服务请求带来的压力,对于大规模还是无力回天

非阻塞IO  (nonblocking IO)

  特点

    基于IO阻塞模型类似,再本应阻塞得地方不在阻塞,如果未收到想要数据会返回一个 error 给用户告知无数据

    发送 error 后会进行其他得任务继续操作背后会一直对 kernel 进行轮询发送数据请求,

    这期间如果数据到了就会重新正常操作,而在轮询期间,无数据得进程会一直处于阻塞状态

    就结果而言。完全得实现了并发,以及解决了IO阻塞带来得效率低下的问题

设置

  1. server.setblocking() #默认是True
  2. server.setblocking(False) #False的话就成非阻塞了,这只是对于socket套接字来说的

不良影响:

    1. 虽说解决了单线程并发,但是大大的占用了cpu

    2.  任务完成的响应延迟增大,任务可能在两次轮询之间的任意时间完成。这会导致整体数据吞吐量的降低

  总结 :

    完全不推荐

多路复用IO  (IO multiplexing)

  特点

    当用户进程调用了select,那么整个进程会被block

    而同时,kernel会“监视”所有select负责的socket,当任何一个socket中的数据准备好了,select就会返回。

    这个时候用户进程再调用read操作,将数据从kernel拷贝到用户进程。

    这个图和blocking IO的图其实并没有太大的不同,事实上还更差一些。

    因为这里需要使用两个系统调用(select和recvfrom),而blocking IO只调用了一个系统调用(recvfrom)。但是,用select的优势在于它可以同时处理多个connection。

设置

  1. 使用 select 模块 或者 eppol (只能用于 linux 中)
    最优的选择方案是用 selectors

    selectors 实例 

  1. #服务端
  2. from socket import *
  3. import selectors
  4.  
  5. sel=selectors.DefaultSelector()
  6. def accept(server_fileobj,mask):
  7. conn,addr=server_fileobj.accept()
  8. sel.register(conn,selectors.EVENT_READ,read)
  9.  
  10. def read(conn,mask):
  11. try:
  12. data=conn.recv(1024)
  13. if not data:
  14. print('closing',conn)
  15. sel.unregister(conn)
  16. conn.close()
  17. return
  18. conn.send(data.upper()+b'_SB')
  19. except Exception:
  20. print('closing', conn)
  21. sel.unregister(conn)
  22. conn.close()
  23.  
  24. server_fileobj=socket(AF_INET,SOCK_STREAM)
  25. server_fileobj.setsockopt(SOL_SOCKET,SO_REUSEADDR,1)
  26. server_fileobj.bind(('127.0.0.1',8088))
  27. server_fileobj.listen(5)
  28. server_fileobj.setblocking(False) #设置socket的接口为非阻塞
  29. sel.register(server_fileobj,selectors.EVENT_READ,accept) #相当于网select的读列表里append了一个文件句柄server_fileobj,并且绑定了一个回调函数accept
  30.  
  31. while True:
  32. events=sel.select() #检测所有的fileobj,是否有完成wait data的
  33. for sel_obj,mask in events:
  34. callback=sel_obj.data #callback=accpet
  35. callback(sel_obj.fileobj,mask) #accpet(server_fileobj,1)
  36.  
  37. #客户端
  38. from socket import *
  39. c=socket(AF_INET,SOCK_STREAM)
  40. c.connect(('127.0.0.1',8088))
  41.  
  42. while True:
  43. msg=input('>>: ')
  44. if not msg:continue
  45. c.send(msg.encode('utf-8'))
  46. data=c.recv(1024)
  47. print(data.decode('utf-8'))

不良影响:

    1. 虽说解决了单线程并发,但是大大的占用了cpu

    2.  任务完成的响应延迟增大,任务可能在两次轮询之间的任意时间完成。这会导致整体数据吞吐量的降低

  总结 :

    select的优势在于可以处理多个连接,不适用于单个连接

异步IO(asynchronous IO)

  用户进程发起read操作之后,立刻就可以开始去做其它的事。

  kernel 受到一个asynchronous read之后,首先它会立刻返回,所以不会对用户进程产生任何block。

  然后,kernel会等待数据准备完成,然后将数据拷贝到用户内存,

  当这一切都完成之后,kernel会给用户进程发送一个signal,告诉它read操作完成了。

socketserver 模块

 包含类

  BaseServer是基类,它不能实例化使用,

  TCPServer使用TCP协议通信,

  UDPServer使用UDP协议通信,

  UnixStreamServer和UnixDatagramServer使用Unix域套接字,只适用于UNIX平台。

  ForkingMixIn 多进程异步

  ThreadingMixIn 多线程异步

如何创建一个socketserver :

  1. 创建一个请求处理的类,继承 BaseRequestHandlerclass ,要重写父类里 handle() 方法;

  2. 你必须实例化 TCPServer,并且传递server IP和你上面创建的请求处理类,给这个TCPServer;

  3. server.handle_requese()  只处理一个请求,server.server_forever() 处理多个一个请求,永远执行

  4. 关闭连接 server_close()

  1. import socketserver
  2.  
  3. class MyTCPHandler(socketserver.BaseRequestHandler): #服务类,监听绑定等等
  4.  
  5. def handle(self): #请求处理类,所有请求的交互都是在handle里执行的
  6. # self.request is the TCP socket connected to the client
  7. self.data = self.request.recv(1024).strip()  #每一个请求都会实例化MyTCPHandler(socketserver.BaseRequestHandler):
  8. print("{} wrote:".format(self.client_address[0]))
  9. print(self.data)
  10. self.request.sendall(self.data.upper())  #sendall是重复调用send.
  11.  
  12. if __name__ == "__main__":
  13. HOST, PORT = "localhost", 9999
  14.  
  15. server = socketserver.TCPServer((HOST, PORT), MyTCPHandler)
  16.  
  17. # Activate the server; this will keep running until you
  18. # interrupt the program with Ctrl-C
  19. server.serve_forever()
  20.  
  21. server = socketserver.ThreadingTCPServer((HOST, PORT), MyTCPHandler) #线程
  22. # server = socketserver.ForkingTCPServer((HOST, PORT), MyTCPHandler) #多进程 linux适用
  23. # server = socketserver.TCPServer((HOST, PORT), MyTCPHandler) 单进程

ThreadingTCPServer

  服务器内部会为每个client创建一个 “线程”,该线程用来和客户端进行交互。

  1. import SocketServer
  2.  
  3. class MyServer(SocketServer.BaseRequestHandler):
  4.  
  5. def handle(self):
  6. pass
  7.  
  8. if __name__ == '__main__':
  9. server = SocketServer.ThreadingTCPServer(('127.0.0.1',8766), MyServer)
  10. server.serve_forever()

ForkingTCPServer

  用法类似 ThreadingTCPServer 只是将线程换成了进程 

  1. import SocketServer
  2.  
  3. class MyServer(SocketServer.BaseRequestHandler):
  4.  
  5. def handle(self):
  6. pass
  7.  
  8. if __name__ == '__main__':
  9. server = SocketServer.ForkingTCPServer(('127.0.0.1',8009),MyServer)
  10. server.serve_forever()

 总结:

socketserver 完美的解决了之前的各种困扰。不论是并发问题还是线程池大小以及IO问题都被解决。

  因此socketserver成为了最终解决方案

4.6 并发编程/IO模型的更多相关文章

  1. python 并发编程 io模型 目录

    python 并发编程 IO模型介绍 python 并发编程 socket 服务端 客户端 阻塞io行为 python 并发编程 阻塞IO模型 python 并发编程 非阻塞IO模型 python 并 ...

  2. Python Web学习笔记之并发编程IO模型

    了解新知识之前需要知道的一些知识 同步(synchronous):一个进程在执行某个任务时,另外一个进程必须等待其执行完毕,才能继续执行 #所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调 ...

  3. Python之并发编程-IO模型

    目录 一.IO模型介绍二.阻塞IO(blocking IO)三.非阻塞IO(non-blocking IO)四.多路复用IO(IO multiplexing)五.异步IO(Asynchronous I ...

  4. 并发编程 - io模型 - 总结

    1.提交任务得方式: 同步:提交完任务,等结果,执行下一个任务 异步:提交完,接着执行,异步 + 回调 异步不等结果,提交完任务,任务执行完后,会自动触发回调函数2.同步不等于阻塞: 阻塞:遇到io, ...

  5. python并发编程&IO模型

    一 IO模型介绍 为了更好地了解IO模型,可先回顾下:同步.异步.阻塞.非阻塞 同步(synchronous) IO和异步(asynchronous) IO,阻塞(blocking) IO和非阻塞(n ...

  6. 并发编程——IO模型

    前言 同步(synchronous):一个进程在执行某个任务时,另外一个进程必须等待其执行完毕,才能继续执行 #所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不会返回.按照这个定义, ...

  7. 并发编程——IO模型详解

    ​ 我是一个Python技术小白,对于我而言,多任务处理一般就借助于多进程以及多线程的方式,在多任务处理中如果涉及到IO操作,则会接触到同步.异步.阻塞.非阻塞等相关概念,当然也是并发编程的基础. ​ ...

  8. 15 并发编程-(IO模型)

    一.IO模型介绍 1.阻塞与非阻塞指的是程序的两种运行状态 阻塞:遇到IO就发生阻塞,程序一旦遇到阻塞操作就会停在原地,并且立刻释放CPU资源 非阻塞(就绪态或运行态):没有遇到IO操作,或者通过某种 ...

  9. 并发编程——IO模型(6)

    1.IO模型分类 同步IO #所谓同步,就是在发出一个功能调用时,在没有得到结果之前,该调用就不会返回.按照这个定义,其实绝大多数函数都是同步调用.但是一般而言,我们在说同步.异步的时候,特指那些需要 ...

随机推荐

  1. 性能测试 基于Python结合InfluxDB及Grafana图表实时采集Linux多主机性能数据

    基于Python结合InfluxDB及Grafana图表实时采集Linux多主机性能数据   by:授客 QQ:1033553122 实现功能 测试环境 环境搭建 使用前提 使用方法 运行程序 效果展 ...

  2. Monkey测试记录

    配置环境变量,不然用不了adb命令 path这里也一样配置一下 命令的各种意思百度一下看看也就知道了 看到一篇博客推荐的一种测试命令,我也直接拿来用了 adb shell monkey -p 你的包名 ...

  3. JPasswordField密码框,JList列表框

    [JPasswordField密码框] //导入Java类 import javax.swing.*; import java.awt.*; import java.awt.event.ActionE ...

  4. Python使用Plotly绘图工具,绘制气泡图

    今天来讲讲如何使用Python 绘图工具,Plotly来绘制气泡图. 气泡图的实现方法类似散点图的实现.修改散点图中点的大小,就变成气泡图. 实现代码如下: import plotly as py i ...

  5. vs2015启动崩溃,wpfgfx_v0400.dll加载D3DCompiler_47.dll失败

    有一段时间没有用vs2015,今天一启动就crash,尝试了以下办法: 1. 卸载掉这段时间安装的一些软件和vs插件 2. 修复安装 3. 卸载.重新安装vs2015 sp2版本 4. devenv. ...

  6. 关于MongoDB时间格式转换和时间段聚合统计的用法总结

    一 . 背景需求 在日常的业务需求中,我们往往会根据时间段来统计数据.例如,统计每小时的下单量:每天的库存变化,这类信息数据对运营管理很重要. 这类数据统计依赖于各个时间维度,年月日.时分秒都有可能. ...

  7. 启动期间的内存管理之build_zonelists初始化备用内存域列表zonelists--Linux内存管理(十三)

    1. 今日内容(第二阶段(二)–初始化备用内存域列表zonelists) 我们之前讲了在memblock完成之后, 内存初始化开始进入第二阶段, 第二阶段是一个漫长的过程, 它执行了一系列复杂的操作, ...

  8. 进程命令(tasklist)

    TaskList命令: // 描述: 显示本地或远程计算机上正在运行的进程列表信息. // 语法: tasklist [/s <computer> [ /u [<domain> ...

  9. 阿里巴巴JAVA开发手册

    Java编程规约 (一)命名风格      1. [强制] 代码中的命名均不能以下划线或美元符号开始,也不能以下划线或美元符号结束.           反例: _name / __name / $O ...

  10. centos7下kubernetes(15。kubernetes-外网访问service)

    kubernetes提供了多种类型的service,默认是cluster IP ClusterIP cluster内部IP对外提供服务,只有cluster内的节点和pod可访问,这是默认的servic ...