帖子中有什么不足之处欢迎大家指正。
python网络编程难的地方不在网络,而在事件反馈机制。
关键要理解什么时候产生事件,以及事件反馈回来结果参数及其含义。
整个过程其实就是注册事件-》产生事件-》返回事件-》处理事件 的整个流程,下面我用老师课上的select例子来描述(通过注释描述)这个过程。
import socket
import select
import Queue
#服务端监听地址和端口
server=('127.0.0.1',2013)
#创建socket
sock=socket.socket(socket.AF_INET,socket.SOCK_STREAM)
#设置socket选项和监听过程
sock.setblocking(False)
sock.setsockopt(socket.SOL_SOCKET,socket.SO_REUSEADDR,1)
sock.bind(server)
sock.listen(10)
#读事件的socket列表,这个很重要,不在这个列表中的socket得不到事件反馈通知,初始只有服务监听socket
rlists=[sock]
#写事件的socket列表
wlists=[]
#接收到的数据队列
msg_que={}
#等待超时20秒
timeout=20
#从后面逻辑看,while rlists相当于while True
while rlists:
"""
select过程就是等待网络事件的过程,分别是读、写和异常事件,这集中了注册事件、产生事件和返回事件的过程
注册事件: 是关注rlists(读),wlists(写),rlists(异常)中的socket列表
产生事件: 当客户端socket连接和发送、服务端向客户端socket写入、以及网络异常这些事件就会分别产生读、写和异常事件
返回事件: 当服务端接收到这些事件就返回相应的列表,分别对应rs,ws,es
如果在timeout时间内没有产生上述事件,也会返回,此时rs,ws,es均为空
"""
rs,ws,es = select.select(rlists,wlists,rlists,timeout)
#这说明没有socket事件
if not(rs or ws or es):
print "timeout......"
continue
#读事件,可能有多个socket产生读事件,s是读事件发生的socket
for s in rs:
#这个很重要,s是服务监听socket,所以是连接请求事件
if s is sock:
#接受客户端socket连接,产生负责和客户端socket进行一对一通讯的socket(就是conn)
conn,addr=s.accept()
print 'connet by ', addr
conn.setblocking(False)
#这个非常重要,如果不把conn加入rlists,那么该socket上的读写事件将得不到通知
rlists.append(conn)
#初始化conn的数据队列,后面读到数据就放入到这个队列中
msg_que[conn]=Queue.Queue()
#这是socket读事件
else:
#读数据
data=s.recv(1024)
#接收到数据
if data:
print data
#将读到的数据放入s的数据队列
msg_que
#将s加入到wlists队列,如果没有这个,s上发生的写事件就得不到事件通知
if s not in wlists:
wlists.append(s)
#没有接收到数据,通常是客户端主动断开socket连接,此时要将s从wlists和rlists中移除,并关闭s,清除s的数据队列
else:
if s in wlists:
wlists.remove(s)
rlists.remove(s)
s.close()
del msg_que
#写事件,可能有多个socket产生写事件,s是写事件发生的socket
for s in ws:
try:
#获取数据队列中的读到的数据
msg=msg_que
except Queue.Empty:
#数据队列中没有数据
print "msg empty"
wlists.remove(s)
else:
#发送数据
s.send(msg)
#异常事件, s是发生异常的socket
for s in es:
print "except",s.getpeername()
#将s从rlists中移除
if s in rlists:
rlists.remove(s)
#将s从wlists中移除
if s in wlists:
wlists.remove(s)
s.close()
del msg_que
sock.close()
print 'server sock closed'
复制代码
poll、epoll流程都是类似的,理解了这个,其它的应该也不在话下
推荐一篇经典e文,我还没来得及看。