在Python中使用异步Socket编程性能测试


Posted in Python onJune 25, 2014

OK,首先写一个python 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的三倍。对客户端的异步改造主要有两点:

同步模式下,客户端分别创建socket;而在异步模式下,client开始就创建了所有的socket。
通过“sock.setblocking(0)”设置socket为异步模式。
通过Unix系统的select俩返回可读取IO
最为核心的是26行和29行。尤其是29行的select操作返回待读取socket的列表。

Python 相关文章推荐
Python实现全角半角转换的方法
Aug 18 Python
用Python程序抓取网页的HTML信息的一个小实例
May 02 Python
使用pdb模块调试Python程序实例
Jun 02 Python
利用Django内置的认证视图实现用户密码重置功能详解
Nov 24 Python
python创建列表和向列表添加元素的实现方法
Dec 25 Python
详解Python最长公共子串和最长公共子序列的实现
Jul 07 Python
详解PyCharm+QTDesigner+PyUIC使用教程
Jun 13 Python
Python pandas DataFrame操作的实现代码
Jun 21 Python
Django如何简单快速实现PUT、DELETE方法
Jul 24 Python
python通过SSH登陆linux并操作的实现
Oct 10 Python
python使用opencv resize图像不进行插值的操作
Jul 05 Python
opencv检测动态物体的实现
Jul 21 Python
Python开发的单词频率统计工具wordsworth使用方法
Jun 25 #Python
python 字典(dict)遍历的四种方法性能测试报告
Jun 25 #Python
用python登录Dr.com思路以及代码分享
Jun 25 #Python
python正则表达式re模块详解
Jun 25 #Python
Python通过websocket与js客户端通信示例分析
Jun 25 #Python
Flask框架学习笔记(一)安装篇(windows安装与centos安装)
Jun 25 #Python
Python中文编码那些事
Jun 25 #Python
You might like
ThinkPHP查询语句与关联查询用法实例
2014/11/01 PHP
smarty内置函数section的用法
2015/01/22 PHP
php在数组中查找指定值的方法
2015/03/17 PHP
总结一些PHP中好用但又容易忽略的小知识
2017/06/02 PHP
php 下 html5 XHR2 + FormData + File API 上传文件操作实例分析
2020/02/28 PHP
Gambit vs CL BO3 第三场 2.13
2021/03/10 DOTA
JS效率个人经验谈(8-15更新),加入range技巧
2007/01/09 Javascript
jquery load()在firefox(火狐)下显示不正常的解决方法
2011/04/05 Javascript
javascript跟随滚动效果插件代码(javascript Follow Plugin)
2013/08/03 Javascript
JS 获取浏览器和屏幕宽高等信息代码
2014/03/31 Javascript
jQuery使用$.get()方法从服务器文件载入数据实例
2015/03/25 Javascript
js创建对象的方法汇总
2016/01/07 Javascript
jQuery移动端图片上传组件
2016/06/12 Javascript
Javascript 普通函数和构造函数的区别
2016/11/05 Javascript
vue货币过滤器的实现方法
2017/04/01 Javascript
xmlplus组件设计系列之列表(4)
2017/04/26 Javascript
微信小程序报错:this.setData is not a function的解决办法
2017/09/27 Javascript
JS中数据结构之栈
2019/01/01 Javascript
JS正则表达式常见函数与用法小结
2020/04/13 Javascript
Openlayers实现扩散的动态点(水纹效果)
2020/08/17 Javascript
[41:52]2018DOTA2亚洲邀请赛3月29日 小组赛A组 TNC VS OpTic
2018/03/30 DOTA
[44:40]Serenity vs Pain 2018国际邀请赛小组赛BO2 第一场 8.19
2018/08/21 DOTA
Scrapy爬虫实例讲解_校花网
2017/10/23 Python
基于Python socket的端口扫描程序实例代码
2018/02/09 Python
详解Python文件修改的两种方式
2019/08/22 Python
Python单元测试工具doctest和unittest使用解析
2019/09/02 Python
Django框架 信号调度原理解析
2019/09/04 Python
python GUI库图形界面开发之PyQt5不规则窗口实现与显示GIF动画的详细方法与实例
2020/03/09 Python
Python实现Word表格转成Excel表格的示例代码
2020/04/16 Python
python 爬取英雄联盟皮肤并下载的示例
2020/12/04 Python
CSS3 rgb and rgba(透明色)的使用详解
2020/09/25 HTML / CSS
自动化毕业生专业自荐书范文
2014/02/04 职场文书
勾股定理课后反思
2014/04/26 职场文书
离婚答辩状范文
2015/05/22 职场文书
小学中队长竞选稿
2015/11/20 职场文书
80后创业总结的9条职场用人思想,记得收藏
2019/08/13 职场文书