异步select

server coding:

  1 #!/usr/bin/python
  2 # -*- coding: utf-8 -*-
  3
  4 import select
  5 import socket
  6 import sys
  7 import Queue
  8
  9 # Create a TCP/IP socket
 10 server = socket.socket(socket.AF_INET,socket.SOCK_STREAM)
 11 server.setblocking(False)
 12
 13 # Bind the socket to the port
 14 server_address = (‘localhost‘,1000)
 15 print >>sys.stderr, ‘starting up on %s port %s‘%server_address
 16 server.bind(server_address)
 17
 18 # Listen for incoming connections
 19 server.listen(5)
 20
 21 # Sockets from which we expect to read
 22 inputs = [server]
 23
 24 # Sockets to which we expect to write
 25 outputs = []
 26
 27 # Outgoing message queues (socket:Queue)
 28 message_queues = {}
 29
 30 while True:
 31     # Wait for at least one of the sockets to be ready for processing
 32     print >>sys.stderr, ‘\nwaiting for the next event‘
 33     #分别筛选出可接收消息的sockets、等待发送消息的sockets、和中途出错的sockets。
 34     #其中exceptional为了把所有可接收消息的sockets都遍历到,所以要从inputs列表中读取。
 35     readable, writable, exceptional = select.select(inputs,outputs,inputs)
 36
 37     # Handle inputs
 38     for s in readable:
 39         #第一种情况,创建一个待接收消息的socket放入inputs
 40         if s is server:
 41             # A "readable" server socket is ready to accept a connection
 42             connection, client_address = s.accept()
 43             print >>sys.stderr, ‘new connection from‘, client_address
 44             connection.setblocking(0)
 45             inputs.append(connection)
 46
 47             # Give the connection a queue for data we want to send
 48             message_queues[connection] = Queue.Queue()
 49
 50         #第二种情况,在后续循环中,已经添加到inputs中的sockets已经不是readable了,所以
 51         #要进行接收消息,消息存在message_queues中,并把该socket添加到outputs。
 52         else:
 53             data = s.recv(1024)
 54             if data:
 55                 # A readable client socket has data
 56                 print >>sys.stderr, ‘receieved "%s" from %s‘%(data.upper(),s.getpeername())
 57                 message_queues[s].put(data)
 58                 # Add output channel for response,添加到待发送消息列表
 59                 if s not in outputs:
 60                     outputs.append(s)
 61
 62             #第三种情况就是这个客户端已经断开了,
 63             #所以你再通过recv()接收到的数据就为空了,
 64             #所以这个时候你就可以把这个跟客户端的连接关闭了。
 65             else:
 66                 print >>sys.stderr, ‘closing‘,client_address,‘after reading no data‘
 67                 if s in outputs:
 68                     outputs.remove(s)#既然客户端都断开了,我就不用再给它返回数据了,所以这时候如果这个客户端的连接对象还在outputs列表中,就把它删掉
 69                 inputs.remove(s)#inputs中也删除掉
 70                 s.close()#把这个连接关闭掉
 71
 72                  # Remove message queue
 73                 del message_queues[s]
 74
 75     # Handle outputs
 76     for s2 in writable:
 77         try:
 78             next_msg = message_queues[s2].get_nowait()
 79         #没有消息了
 80         except Queue.Empty:
 81             # No messages waiting so stop checking for writability.
 82             print >>sys.stderr,‘output queue for‘,s2.getpeername(),‘is empty‘
 83             outputs.remove(s2)
 84         #发送消息
 85         else:
 86             print >>sys.stderr, ‘sending "%s" to %s‘%(next_msg,s2.getpeername())
 87             s2.send(next_msg)
 88
 89     # Handle "exceptional conditions"
 90     for s3 in exceptional:
 91         print >>sys.stderr, ‘handling exceptional condition for ‘,s3.getpeername()
 92         # Stop listening for input on the connection
 93         inputs.remove(s3)
 94         if s3 in outputs:
 95             outputs.remove(s3)
 96         s3.close()
 97
 98         #Remove message queue
 99         del message_queues[s3]
100
101
102             

client coding

 1 # -*- coding: utf-8 -*-
 2 """
 3 Created on Sun Oct 23 14:49:20 2016
 4
 5 @author: fuzzier
 6 """
 7
 8 import sys
 9 import socket
10
11 messages = [‘This is the message.‘,
12             ‘It will be sent‘,
13             ‘in parts.‘
14             ]
15 server_address = (‘localhost‘,1000)
16
17 # Create a TCP/IP socket,创建两个客户端
18 socks = [socket.socket(socket.AF_INET,socket.SOCK_STREAM),
19          socket.socket(socket.AF_INET,socket.SOCK_STREAM)
20          ]
21
22 # Connect the socket to the port where the server is listening
23 print >>sys.stderr, ‘connecting to %s port %s‘%server_address
24 for s in socks:
25     s.connect(server_address)
26
27 for message in messages:
28
29     # Send messages on both sockets
30     for s2 in socks:
31         print >>sys.stderr, ‘%s:sending "%s"‘%(s2.getsockname(),message)
32         s2.send(message)
33
34     #Read responses on both sockets
35     for s3 in socks:
36         data = s3.recv(1024)
37         print >>sys.stderr, ‘%s: received "%s"‘ % (s3.getsockname(), data.upper())
38         if not data:
39             print >>sys.stderr,‘closing socket‘,s3.getsockname()
40             s3.close()

原理图:

运行结果:

时间: 2024-10-25 23:27:29

异步select的相关文章

[Z] linux基础编程:IO模型:阻塞/非阻塞/IO复用 同步/异步 Select/Epoll/AIO

原文链接:http://blog.csdn.net/colzer/article/details/8169075 IO概念 Linux的内核将所有外部设备都可以看做一个文件来操作.那么我们对与外部设备的操作都可以看做对文件进行操作.我们对一个文件的读写,都通过调用内核提供的系统调用:内核给我们返回一个file descriptor(fd,文件描述符).而对一个socket的读写也会有相应的描述符,称为socketfd(socket描述符).描述符就是一个数字,指向内核中一个结构体(文件路径,数据

linux基础编程:IO模型:阻塞/非阻塞/IO复用 同步/异步 Select/Epoll/AIO(转载)

IO概念 Linux的内核将所有外部设备都可以看做一个文件来操作.那么我们对与外部设备的操作都可以看做对文件进行操作.我们对一个文件的读写,都通过调用内核提供的系统调用:内核给我们返回一个file descriptor(fd,文件描述符).而对一个socket的读写也会有相应的描述符,称为socketfd(socket描述符).描述符就是一个数字,指向内核中一个结构体(文件路径,数据区,等一些属性).那么我们的应用程序对文件的读写就通过对描述符的读写完成. linux将内存分为内核区,用户区.l

python 异步 select pooll epoll

概念: 首先列一下,sellect.poll.epoll三者的区别 select select最早于1983年出现在4.2BSD中,它通过一个select()系统调用来监视多个文件描述符的数组,当select()返回后,该数组中就绪的文件描述符便会被内核修改标志位,使得进程可以获得这些文件描述符从而进行后续的读写操作. select目前几乎在所有的平台上支持,其良好跨平台支持也是它的一个优点,事实上从现在看来,这也是它所剩不多的优点之一. select的一个缺点在于单个进程能够监视的文件描述符的

【学院官方整理】Python学习路线图-适合自学者从入门到项目开发(视频+文档) 干货提炼

亲爱的学员们: 您好!51CTO学院为梦想增值,诚邀您的关注!51CTO学院致力于让专家分享技术并让技术变现,让广大技术爱好者便捷.实惠的获取优质学习资源. 为了方便大家的学习,我们特别整理了Python学习路线图-适合自学者从入门到项目开发(视频+文档) 干货提炼,点击查看更多>> Python学习路线图-适合自学者从入门到项目开发(视频+文档) 干货提炼 本专题涵盖了Python基础.网络编程.WEB开发基础.WEB框架.分布式监控开发.审计堡垒机.FTP服务器.CMDB.主机批量管理.W

python 开发一个支持多用户在线的FTP

### 作者介绍:* author:lzl### 博客地址:* http://www.cnblogs.com/lianzhilei/p/5813986.html### 功能实现 作业:开发一个支持多用户在线的FTP程序 要求: 用户加密认证 允许同时多用户登录 每个用户有自己的家目录 ,且只能访问自己的家目录 对用户进行磁盘配额,每个用户的可用空间不同 允许用户在ftp server上随意切换目录 允许用户查看当前目录下文件 允许上传和下载文件,保证文件一致性 文件传输过程中显示进度条 附加功能

socket阻塞与非阻塞,同步与异步、I/O模型,select与poll、epoll比较

1. 概念理解 在进行网络编程时,我们常常见到同步(Sync)/异步(Async),阻塞(Block)/非阻塞(Unblock)四种调用方式: 同步/异步主要针对C端: 同步:      所谓同步,就是在c端发出一个功能调用时,在没有得到结果之前,该调用就不返回.也就是必须一件一件事做,等前一件做完了才能做下一件事. 例如普通B/S模式(同步):提交请求->等待服务器处理->处理完毕返回 这个期间客户端浏览器不能干任何事 异步:      异步的概念和同步相对.当c端一个异步过程调用发出后,调

python第五十三天--进程,协程.select.异步I/O...

进程: 1 #!usr/bin/env python 2 #-*-coding:utf-8-*- 3 # Author calmyan 4 import multiprocessing,threading,time 5 6 def run(name): 7 t=threading.Thread(target=run2)#创建新线程 8 t.start() 9 print('进程[%s],打印中...'%name) 10 time.sleep(1) 11 12 def run2(): 13 pri

转一贴,今天实在写累了,也看累了--【Python异步非阻塞IO多路复用Select/Poll/Epoll使用】

下面这篇,原理理解了, 再结合 这一周来的心得体会,整个框架就差不多了... http://www.haiyun.me/archives/1056.html 有许多封装好的异步非阻塞IO多路复用框架,底层在linux基于最新的epoll实现,为了更好的使用,了解其底层原理还是有必要的.下面记录下分别基于Select/Poll/Epoll的echo server实现.Python Select Server,可监控事件数量有限制: 1 2 3 4 5 6 7 8 9 10 11 12 13 14

structs2注解+jsp+ajax实现post异步加载select

流程: 1.点击加载btn发起异步请求post 2.后台处理请求返回数据 3.前端获取数据成功,对数据进行处理 前端: html:首先要导入jq包,不然怎么用ajax呢. <script type="text/javascript" src="js/jquery.js"></script> <select name="warId" id="warId" style="height:100