服务器端编程经常需要构造高性能的IO模型,常见的IO模型有四种:

(1)同步阻塞IO(Blocking IO):即传统的IO模型。

(2)同步非阻塞IO(Non-blocking IO):默认创建的socket都是阻塞的,非阻塞IO要求socket被设置为NONBLOCK。注意这里所说的NIO并非Java的NIO(New IO)库。

(3)IO多路复用(IO Multiplexing):即经典的Reactor设计模式,有时也称为异步阻塞IO,Java中的Selector和Linux中的epoll都是这种模型。

(4)异步IO(Asynchronous IO):即经典的Proactor设计模式,也称为异步非阻塞IO。

同步和异步的概念描述的是用户线程与内核的交互方式:同步是指用户线程发起IO请求后需要等待或者轮询内核IO操作完成后才能继续执行;而异步是指用户线程发起IO请求后仍继续执行,当内核IO操作完成后会通知用户线程,或者调用用户线程注册的回调函数。

阻塞和非阻塞的概念描述的是用户线程调用内核IO操作的方式:阻塞是指IO操作需要彻底完成后才返回到用户空间;而非阻塞是指IO操作被调用后立即返回给用户一个状态值,无需等到IO操作彻底完成。

详细了参考转载的IO多路复用机制详解

IO多路复用

I/O多路复用指:通过一种机制,可以监视多个描述符,一旦某个描述符就绪(一般是读就绪或者写就绪),能够通知程序进行相应的读写操作。

Linux

Linux中的 select,poll,epoll 都是IO多路复用的机制。他们之间的区别参考另一篇博文 http://www.cnblogs.com/wjx1/p/5082640.html

Python

Python中有一个select模块,其中提供了:select、poll、epoll三个方法,分别调用系统的 select,poll,epoll 从而实现IO多路复用。

  1. Windows Python:
  2.     提供: select
  3. Mac Python:
  4.     提供: select
  5. Linux Python:
  6.     提供: select、poll、epoll

注意:网络操作、文件操作、终端操作等均属于IO操作,对于windows只支持Socket操作,其他系统支持其他IO操作,但是无法检测 普通文件操作 自动上次读取是否已经变化。

SocketServer模块

SocketServer内部使用 IO多路复用 以及 “多线程” 和 “多进程” ,从而实现并发处理多个客户端请求的Socket服务端。即:每个客户端请求连接到服务器时,Socket服务端都会在服务器是创建一个“线程”或者“进 程” 专门负责处理当前客户端的所有请求。

ThreadingTCPServer

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

1、ThreadingTCPServer基础

使用ThreadingTCPServer:

  • 创建一个继承自 SocketServer.BaseRequestHandler 的类
  • 类中必须定义一个名称为 handle 的方法
  • 启动ThreadingTCPServer
  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. import SocketServer
  4.  
  5. class MyServer(SocketServer.BaseRequestHandler):
  6.  
  7. def handle(self):
  8. # print self.request,self.client_address,self.server
  9. conn = self.request
  10. conn.sendall('欢迎致电 10086,请输入1xxx,0转人工服务.')
  11. Flag = True
  12. while Flag:
  13. data = conn.recv(1024)
  14. if data == 'exit':
  15. Flag = False
  16. elif data == '':
  17. conn.sendall('通过可能会被录音.balabala一大推')
  18. else:
  19. conn.sendall('请重新输入.')
  20.  
  21. if __name__ == '__main__':
  22. server = SocketServer.ThreadingTCPServer(('127.0.0.1',8009),MyServer)
  23. server.serve_forever()
  24.  
  25. SocketServer实现服务器

SocketServer实现服务器

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. import socket
  5.  
  6. ip_port = ('127.0.0.1',8009)
  7. sk = socket.socket()
  8. sk.connect(ip_port)
  9. sk.settimeout(5)
  10.  
  11. while True:
  12. data = sk.recv(1024)
  13. print 'receive:',data
  14. inp = raw_input('please input:')
  15. sk.sendall(inp)
  16. if inp == 'exit':
  17. break
  18.  
  19. sk.close()
  20.  
  21. 客户端

客户端

2、ThreadingTCPServer源码剖析

ThreadingTCPServer的类图关系如下:

内部调用流程为:

  • 启动服务端程序
  • 执行 TCPServer.__init__ 方法,创建服务端Socket对象并绑定 IP 和 端口
  • 执行 BaseServer.__init__ 方法,将自定义的继承自SocketServer.BaseRequestHandler 的类 MyRequestHandle赋值给 self.RequestHandlerClass
  • 执行 BaseServer.server_forever 方法,While 循环一直监听是否有客户端请求到达 ...
  • 当客户端连接到达服务器
  • 执行 ThreadingMixIn.process_request 方法,创建一个 “线程” 用来处理请求
  • 执行 ThreadingMixIn.process_request_thread 方法
  • 执行 BaseServer.finish_request 方法,执行 self.RequestHandlerClass()  即:执行 自定义 MyRequestHandler 的构造方法(自动调用基类BaseRequestHandler的构造方法,在该构造方法中又会调用 MyRequestHandler的handle方法)

实例:

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. import SocketServer
  4.  
  5. class MyServer(SocketServer.BaseRequestHandler):
  6.  
  7. def handle(self):
  8. # print self.request,self.client_address,self.server
  9. conn = self.request
  10. conn.sendall('欢迎致电 10086,请输入1xxx,0转人工服务.')
  11. Flag = True
  12. while Flag:
  13. data = conn.recv(1024)
  14. if data == 'exit':
  15. Flag = False
  16. elif data == '':
  17. conn.sendall('通过可能会被录音.balabala一大推')
  18. else:
  19. conn.sendall('请重新输入.')
  20.  
  21. if __name__ == '__main__':
  22. server = SocketServer.ThreadingTCPServer(('127.0.0.1',8009),MyServer)
  23. server.serve_forever()
  24.  
  25. 服务端

服务端

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. import socket
  5.  
  6. ip_port = ('127.0.0.1',8009)
  7. sk = socket.socket()
  8. sk.connect(ip_port)
  9. sk.settimeout(5)
  10.  
  11. while True:
  12. data = sk.recv(1024)
  13. print 'receive:',data
  14. inp = raw_input('please input:')
  15. sk.sendall(inp)
  16. if inp == 'exit':
  17. break
  18.  
  19. sk.close()
  20.  
  21. 客户端

客户端

源码精简:

  1. import socket
  2. import threading
  3. import select
  4.  
  5. def process(request, client_address):
  6. print request,client_address
  7. conn = request
  8. conn.sendall('欢迎致电 10086,请输入1xxx,0转人工服务.')
  9. flag = True
  10. while flag:
  11. data = conn.recv(1024)
  12. if data == 'exit':
  13. flag = False
  14. elif data == '':
  15. conn.sendall('通过可能会被录音.balabala一大推')
  16. else:
  17. conn.sendall('请重新输入.')
  18.  
  19. sk = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
  20. sk.bind(('127.0.0.1',8002))
  21. sk.listen(5)
  22.  
  23. while True:
  24. r, w, e = select.select([sk,],[],[],1)
  25. print 'looping'
  26. if sk in r:
  27. print 'get request'
  28. request, client_address = sk.accept()
  29. t = threading.Thread(target=process, args=(request, client_address))
  30. t.daemon = False
  31. t.start()
  32.  
  33. sk.close()

如精简代码可以看出,SocketServer的ThreadingTCPServer之所以可以同时处理请求得益于 selectThreading 两个东西,其实本质上就是在服务器端为每一个客户端创建一个线程,当前线程用来处理对应客户端的请求,所以,可以支持同时n个客户端链接(长连接)。

ForkingTCPServer

ForkingTCPServer和ThreadingTCPServer的使用和执行流程基本一致,只不过在内部分别为请求者建立 “线程”  和 “进程”。

基本使用:

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3. import SocketServer
  4.  
  5. class MyServer(SocketServer.BaseRequestHandler):
  6.  
  7. def handle(self):
  8. # print self.request,self.client_address,self.server
  9. conn = self.request
  10. conn.sendall('欢迎致电 10086,请输入1xxx,0转人工服务.')
  11. Flag = True
  12. while Flag:
  13. data = conn.recv(1024)
  14. if data == 'exit':
  15. Flag = False
  16. elif data == '':
  17. conn.sendall('通过可能会被录音.balabala一大推')
  18. else:
  19. conn.sendall('请重新输入.')
  20.  
  21. if __name__ == '__main__':
  22. server = SocketServer.ForkingTCPServer(('127.0.0.1',8009),MyServer)
  23. server.serve_forever()
  24.  
  25. 服务端

服务端

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. import socket
  5.  
  6. ip_port = ('127.0.0.1',8009)
  7. sk = socket.socket()
  8. sk.connect(ip_port)
  9. sk.settimeout(5)
  10.  
  11. while True:
  12. data = sk.recv(1024)
  13. print 'receive:',data
  14. inp = raw_input('please input:')
  15. sk.sendall(inp)
  16. if inp == 'exit':
  17. break
  18.  
  19. sk.close()
  20.  
  21. 客户端

客户端

以上ForkingTCPServer只是将 ThreadingTCPServer 实例中的代码:

  • server = SocketServer.ThreadingTCPServer(('127.0.0.1',8009),MyRequestHandler)
  • 变更为
  • server =SocketServer.ForkingTCPServer(('127.0.0.1',8009),MyRequestHandler)

SocketServer的ThreadingTCPServer之所以可以同时处理请求得益于 select 和 os.fork 两个东西,其实本质上就是在服务器端为每一个客户端创建一个进程,当前新创建的进程用来处理对应客户端的请求,所以,可以支持同时n个客户端链接(长连接)。

源码剖析参考 ThreadingTCPServer

Twisted

Twisted是一个事件驱动的网络框架,其中包含了诸多功能,例如:网络协议、线程、数据库管理、网络操作、电子邮件等。

事件驱动

简而言之,事件驱动分为二个部分:第一,注册事件;第二,触发事件。

自定义事件驱动框架,命名为:“弑君者”:

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. # event_drive.py
  5.  
  6. event_list = []
  7.  
  8. def run():
  9. for event in event_list:
  10. obj = event()
  11. obj.execute()
  12.  
  13. class BaseHandler(object):
  14. """
  15. 用户必须继承该类,从而规范所有类的方法(类似于接口的功能)
  16. """
  17. def execute(self):
  18. raise Exception('you must overwrite execute')
  19.  
  20. 最牛逼的事件驱动框架

程序员使用“弑君者框架”:

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. from source import event_drive
  5.  
  6. class MyHandler(event_drive.BaseHandler):
  7.  
  8. def execute(self):
  9. print 'event-drive execute MyHandler'
  10.  
  11. event_drive.event_list.append(MyHandler)
  12. event_drive.run()

如上述代码,事件驱动只不过是框架规定了执行顺序,程序员在使用框架时,可以向原执行顺序中注册“事件”,从而在框架执行时可以出发已注册的“事件”。

基于事件驱动Socket

  1. #!/usr/bin/env python
  2. # -*- coding:utf-8 -*-
  3.  
  4. from twisted.internet import protocol
  5. from twisted.internet import reactor
  6.  
  7. class Echo(protocol.Protocol):
  8. def dataReceived(self, data):
  9. self.transport.write(data)
  10.  
  11. def main():
  12. factory = protocol.ServerFactory()
  13. factory.protocol = Echo
  14.  
  15. reactor.listenTCP(8000,factory)
  16. reactor.run()
  17.  
  18. if __name__ == '__main__':
  19. main()

程序执行流程:

  • 运行服务端程序
  • 创建Protocol的派生类Echo
  • 创建ServerFactory对象,并将Echo类封装到其protocol字段中
  • 执行reactor的 listenTCP 方法,内部使用 tcp.Port 创建socket server对象,并将该对象添加到了 reactor的set类型的字段 _read 中
  • 执行reactor的 run 方法,内部执行 while 循环,并通过 select 来监视 _read 中文件描述符是否有变化,循环中...
  • 客户端请求到达
  • 执行reactor的 _doReadOrWrite 方法,其内部通过反射调用 tcp.Port 类的 doRead 方法,内部 accept 客户端连接并创建Server对象实例(用于封装客户端socket信息)和 创建 Echo 对象实例(用于处理请求) ,然后调用 Echo 对象实例的 makeConnection 方法,创建连接。
  • 执行 tcp.Server 类的 doRead 方法,读取数据,
  • 执行 tcp.Server 类的 _dataReceived 方法,如果读取数据内容为空(关闭链接),否则,出发 Echo 的 dataReceived 方法
  • 执行 Echo 的 dataReceived 方法

从源码可以看出,上述实例本质上使用了事件驱动的方法 和 IO多路复用的机制来进行Socket的处理。

python--第十天总结(IO多路复用)的更多相关文章

  1. python基础-11 socket,IO多路复用,select伪造多线程,select读写分离。socketserver源码分析

    Socket socket通常也称作"套接字",用于描述IP地址和端口,是一个通信链的句柄,应用程序通常通过"套接字"向网络发出请求或者应答网络请求. sock ...

  2. Python之路--协程/IO多路复用

    引子: 之前学习过了,线程,进程的概念,知道了在操作系统中进程是资源分配的最小单位,线程是CPU调度的最小单位.按道理来说我们已经算是把CPU的利用率提高很多了.但是我们知道无论是创建多进程还是创建多 ...

  3. IO多路复用select/poll/epoll详解以及在Python中的应用

    IO multiplexing(IO多路复用) IO多路复用,有些地方称之为event driven IO(事件驱动IO). 它的好处在于单个进程可以处理多个网络IO请求.select/epoll这两 ...

  4. IO多路复用,协程

    https://www.cnblogs.com/wangjun187197/p/9642429.html Python之路--协程/IO多路复用 I/O复用模型 此模型用到select和poll函数, ...

  5. python运维开发(十)----IO多路复用线程基本使用

    内容目录: python作用域 python2.7和python3.5的多继承区别 IO多路复用 socketserver模块源分析 多线程.进程.协程 python作用域  python中无块级作用 ...

  6. python第十周:进程、协程、IO多路复用

    多进程(multiprocessing): 多进程的使用 multiprocessing是一个使用类似于线程模块的API支持产生进程的包. 多处理包提供本地和远程并发,通过使用子进程而不是线程有效地侧 ...

  7. Python(七)Socket编程、IO多路复用、SocketServer

    本章内容: Socket IO多路复用(select) SocketServer 模块(ThreadingTCPServer源码剖析) Socket socket通常也称作"套接字" ...

  8. python中的IO多路复用

    在python的网络编程里,socetserver是个重要的内置模块,其在内部其实就是利用了I/O多路复用.多线程和多进程技术,实现了并发通信.与多进程和多线程相比,I/O多路复用的系统开销小,系统不 ...

  9. socket_server源码剖析、python作用域、IO多路复用

    本节内容: 课前准备知识: 函数嵌套函数的使用方法: 我们在使用函数嵌套函数的时候,是学习装饰器的时候,出现过,由一个函数返回值是一个函数体情况. 我们在使用函数嵌套函数的时候,最好也这么写. def ...

  10. python之IO多路复用

    在python的网络编程里,socetserver是个重要的内置模块,其在内部其实就是利用了I/O多路复用.多线程和多进程技术,实现了并发通信.与多进程和多线程相比,I/O多路复用的系统开销小,系统不 ...

随机推荐

  1. RabbitMq入门以及使用教程

    祭出原帖:https://blog.csdn.net/lyhkmm/article/details/78772919 原文转载:http://blog.csdn.net/whycold/article ...

  2. Python搭建环境

    小白开启Python之门啦啦啦!!!!! 学习任何一种语言,第一步就是环境的搭建,小白python之旅开始啦!!!加油加油,抬头挺胸齐步走~~~ 目前大家开发系统主要是,windows.Mac OS ...

  3. 《Linux内核原理与分析》第六周作业

    课本:第五章 系统调用的三层机制(下) 中断向量0x80和system_call中断服务程序入口的关系 0x80对应着system_call中断服务程序入口,在start_kernel函数中调用了tr ...

  4. Redis集群配置(linux)

     *弄了一天,有问题直接问我.qq:137416943   1.redis集群的配置和简单使用   Redis集群配置 0.首先要配置环境: 0.1 安装c++ yum install gcc-c++ ...

  5. python中str函数isdigit、isdecimal、isnumeric的区别

    num = "1" #unicodenum.isdigit() # Truenum.isdecimal() # Truenum.isnumeric() # True num = & ...

  6. 【java多线程】队列系统之说说队列Queue

    转载:http://benjaminwhx.com/2018/05/05/%E8%AF%B4%E8%AF%B4%E9%98%9F%E5%88%97Queue/ 1.简介 Queue(队列):一种特殊的 ...

  7. 用 Excel 画正态分布曲线

    用 Excel 画正态分布曲线 群里有小伙伴询问一道曲线题,有小伙伴看出来是正态分布曲线,刚好之前有大概了解一下正态分布. 可以在 Excel 很容易实现. 使用 NORMDIST 即可. 效果如下:

  8. 查找二叉树(tree_a)

    问题 E: 查找二叉树(tree_a) 时间限制: 1 Sec  内存限制: 128 MB提交: 206  解决: 152[提交][状态][讨论版][命题人:quanxing][Edit] [Test ...

  9. Python中Lambda表达式使用

    软件环境 Python: 2.7.13; win10 Lambda描述 python 使用 lambda 表达式来创建匿名函数 lambda只是一个表达式,函数体比def简单很多 lambda的主体是 ...

  10. ThinkPHP5.*版本发布安全更新

    2018 年 12 月 9 日 发布 本次版本更新主要涉及一个安全更新,由于框架对控制器名没有进行足够的检测会导致在没有开启强制路由的情况下可能的getshell漏洞,受影响的版本包括5.0和5.1版 ...