浏览 6676 次
锁定老帖子 主题:Python异步网络探索
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
|
|
---|---|
作者 | 正文 |
发表时间:2013-03-26
最后修改:2013-03-26
OK,首先写一个pythone socket的server段,对开放三个端口:10000,10001,10002.krondo的例子中是每个server绑定一个端口,测试的时候需要分别开3个shell,分别运行.这太麻烦了,就分别用三个Thread来运行这些services. import optparse import os import socket import time from threading import Thread import StringIO txt = '''1111 2222 3333 4444 ''' def server(listen_socket): while True: buf = StringIO.StringIO(txt) sock, addr = listen_socket.accept() print 'Somebody at %s wants poetry!' % (addr,) while True: try: line = buf.readline().strip() if not line: sock.close() break sock.sendall(line) # this is a blocking call print 'send bytes to client:%s' % line #sock.close() except socket.error: sock.close() break time.sleep(1) #server和client连接后,server会故意每发送一个单词后等待一秒钟后再发送另一个单词 def main(): ports = [10000, 10001, 10002] for port in ports: listen_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM) listen_socket.setsockopt(socket.SOL_SOCKET, socket.SO_REUSEADDR, 1) addres = (str('127.0.0.1'), port) listen_socket.bind(addres) listen_socket.listen(5) print "start listen at:%s" % (port,) worker = Thread(target = server, args = [listen_socket]) worker.setDaemon(True) worker.start() if __name__ == '__main__': main() while True: time.sleep(0.1) #如果不sleep的话,CPU会被Python完全占用了 pass 下面是一个client,没有才用异步网络,连接这个三个端口的server: import socket if __name__ == '__main__': ports = [10000, 10001, 10002] for port in ports: address = (str('127.0.0.1'), port) sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.connect(address) poem = '' while True: data = sock.recv(4) if not data: sock.close() break poem += data print poem 下面用异步的client来读取,代码如下: import datetime, errno, optparse, select, socket def connect(port): """Connect to the given server and return a non-blocking socket.""" address = (str('127.0.0.1'), port) sock = socket.socket(socket.AF_INET, socket.SOCK_STREAM) sock.connect(address) sock.setblocking(0) return sock def format_address(address): host, port = address return '%s:%s' % (host or '127.0.0.1', port) if __name__ == '__main__': ports = [10000, 10001, 10002] start = datetime.datetime.now() sockets = map(connect, ports) poems = dict.fromkeys(sockets, '') # socket -> accumulated poem # socket -> task numbers sock2task = dict([(s, i + 1) for i, s in enumerate(sockets)]) sockets = list(sockets) # make a copy while sockets: #运用select来确保那些可读取的异步socket可以立即开始读取IO #OS不停的搜索目前可以read的socket,有的话就返回rlist rlist, _, _ = select.select(sockets, [], []) for sock in rlist: data = '' while True: try: new_data = sock.recv(1024) except socket.error, e: if e.args[0] == errno.EWOULDBLOCK: break raise else: if not new_data: break else: print new_data data += new_data task_num = sock2task[sock] if not data: sockets.remove(sock) sock.close() print 'Task %d finished' % task_num else: addr_fmt = format_address(sock.getpeername()) msg = 'Task %d: got %d bytes of poetry from %s' print msg % (task_num, len(data), addr_fmt) poems[sock] += data elapsed = datetime.datetime.now() - start print 'Got poems in %s' % elapsed 结果只需要4秒就完成了读取任务。效率是刚才同步socket的三倍。对客户端的异步改造主要有两点:
声明:ITeye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
推荐链接
|
|
返回顶楼 | |
发表时间:2013-03-27
python的SimpleHTTPServer就使用了select方法
select使用比较简便,但是在稍大并发的情况下就不行了 现在一般使用epoll来做异步架构的服务端,可支持较大并发。 python有twisted框架,java的nio,netty,mina等框架。 他们在linux上都是epoll的包装 还有aio,现在还不是太成熟就不说了 |
|
返回顶楼 | |
发表时间:2013-03-27
你这个服务端哪里异步了?
|
|
返回顶楼 | |
发表时间:2013-03-27
池中物 写道 你这个服务端哪里异步了?
服务端没有异步,用三个线程; 客户端用异步 |
|
返回顶楼 | |
发表时间:2013-03-29
select需要从用户空间拷贝数据到内核空间,在fd少的情况下还行,多了相对于用空间换时间的epoll性能下降就太快了
|
|
返回顶楼 | |
发表时间:2013-04-02
select在短连接的时候性能还是很好的,epoll其实从底层上说也是用的轮询。
|
|
返回顶楼 | |
发表时间:2013-04-03
panggezi 写道 select在短连接的时候性能还是很好的,epoll其实从底层上说也是用的轮询。
epoll是通知,不是轮循。使用时需要注册你关注的fd上发生的事件 |
|
返回顶楼 | |
发表时间:2013-04-07
jakieyoung 写道 panggezi 写道 select在短连接的时候性能还是很好的,epoll其实从底层上说也是用的轮询。
epoll是通知,不是轮循。使用时需要注册你关注的fd上发生的事件 select,epoll可以参考下面的解释,epoll_wait也是在轮询, http://blog.csdn.net/vividonly/article/details/7539342 |
|
返回顶楼 | |