asyncio与gevent并发性能测试
asyncio与gevent并发性能测试
在对网站进行扫描或者暴破时需要对网站进行高并发操作,然而
requests+concurrent多线程性能上不太理想,了解到python用得比较多的并发库有asyncio和gevent,于是就有了如下测试。
0x00 协程
asyncio和gevent都是基于携程来进行并发操作的。协程也被称为微线程。
协程只是在单一的线程里进行切换不同的协程,因此无法使用多CPU能力,对于CPU密集型程序还是使用多进程比较好。
协程相比较进程和线程来说占用的内容更少,同样的线程切换更多的是靠操作系统来控制,而协程的执行则由我们自己控制。
并发原理:当其中一个协程遇到io等待时,将会切换到另一个协程继续运行。
0x01 grequests
grequests是对requests和gevent库的封装
测试代码:
#!/usr/bin/python3.7
import grequests
import time
if __name__ == '__main__':
start = time.time()
greenlets = []
for _ in range(10):
greenlets.append(grequests.get("http://150.xx.xx.xx"))
rets = grequests.map(greenlets)
for ret in rets:
print(ret)
end = time.time()
print("grequests visit_async tasks %.2f seconds" % (end - start))
grequests.map()参数说明:def grequests.map(requests, stream=False, size=None, exception_handler=None, gtimeout=None)
| 参数 | 说明 | 备注 |
|---|---|---|
| size | 协程的并发度(相当于线程数) | 当一个协程在IO等待时,会将CPU交给其他协程 |
| exception_handler | 异常处理函数 | 用于处理单个请求出现异常的函数 |
| gtimeout | 设置所有请求的超时时间 |
grequests的底层是request,所以它也支持回调函数:
def print_url(r, *args, **kwargs):
print(r.url)
res = grequests.get(url, callback=print_url)
测试结果:

0x02 asyncio + uvloop
由于gevent的猴子补丁的缘故,requests可以和gevent结合使用,但是在不清楚内部实现的情况下,requests库经常比较容易出现Failed to establish a new connection:的情况,在使用grequests库之后该情况得到解决。
uvloop是用Cython写的,目前不支持windows,它基于libuv.uvloop使得asyncio更快,基于性能的测试接近于go。
可以通过两种方式来使用uvloop:
import uvloop
import asyncio
#1. 通过设置策略
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
#2. 直接创建一个新的event_loop
asyncio.set_event_loop(uvloop.new_event_loop())
由于asycnio采用异步操作,它在使用的过程中所有的模块也都得是异步的,所以在进行http请求时也需要异步,即aiohttp
测试代码:
#!/usr/bin/python3.7
import asyncio
import aiohttp
import uvloop
import time
async def access_url(url):
async with aiohttp.ClientSession(connector=aiohttp.TCPConnector(ssl=False)) as session:
async with session.get(url) as response:
status_code = response.status
print(status_code)
async def visit_async():
start = time.time()
tasks = []
for _ in range(10):
tasks.append(access_url("http://150.xx.xx.xx"))
await asyncio.gather(*tasks)
end = time.time()
print("asyncio visit_async tasks %.2f seconds" % (end - start))
if __name__ == '__main__':
loop = asyncio.get_event_loop()
future = asyncio.ensure_future(visit_async())
asyncio.set_event_loop_policy(uvloop.EventLoopPolicy())
loop.run_until_complete(future)
测试结果:

0x03 优缺点
asyncio由于是异步操作,且代码库生态不够完善,部分异步代码库存在问题可能查不到,且编写代码时行数较多,影响阅读,而且代库函数全部重构,上手有难度,但是并发执行的速度较快,对于暴破、端口扫描等比较适用。gevent采用了requests模块,在使用了猴子补丁后对于扫描网站路径等可以有效即时针对扫描结果进行深层扫描。
请求内容:

参考:
- https://www.cnblogs.com/kxsph/p/9268774.html
- http://www.cnblogs.com/zhaof/p/7536569.html
- https://www.cnblogs.com/thomson-fred/p/10142226.html
- asyncio port scanner: https://gist.github.com/0xpizza/dd5e005a0efeb1edfc939d3a409e22d9
asyncio与gevent并发性能测试的更多相关文章
- ORM增删改查并发性能测试
这两天在对一些ORM进行性能测试(涉及SqlSugar.FreeSql.Fast.Framework.Dapper.LiteSql),测试用的是Winform程序,别人第一眼看到我的程序,说,你这测试 ...
- ORM增删改查并发性能测试2
前言 上一篇<ORM增删改查并发性能测试>出现了点小失误,有的输出SQL日志的代码没有禁用,数据库连接字符串可能有问题.统一环境,统一代码后,重新写一篇. 这次重点是并发性能测试,真不是为 ...
- python异步编程之asyncio(百万并发)
前言:python由于GIL(全局锁)的存在,不能发挥多核的优势,其性能一直饱受诟病.然而在IO密集型的网络编程里,异步处理比同步处理能提升成百上千倍的效率,弥补了python性能方面的短板,如最 ...
- jmeter如何玩之badboy + jmeter并发性能测试
今天下班时公司安排了一个同事来对项目做集群性能测试,怀着对性能测试的好奇心,下班后没有着急离开,而是等待 那位同事的到来,然后在旁边学习了下如何使用Badboy和jmeter做性能测试. 1. 软件介 ...
- 流畅的python第十八章使用asyncio包处理并发
对比一个简单的多线程程序和对应的 asyncio 版,说明多线程和异步任务之间的关系asyncio.Future 类与 concurrent.futures.Future 类之间的区别摒弃线程或进程, ...
- python 携程asyncio 实现高并发示例2
https://www.bilibili.com/video/BV1g7411k7MD?from=search&seid=13649975876676293013 import asyncio ...
- 30行代码搞定WCF并发性能测试
[以下只是个人观点,欢迎交流] 30行代码搞定WCF并发性能 轻量级测试. 1. 调用并发测试接口 static void Main() { List< ...
- 如何使用jMeter对某个OData服务进行高并发性能测试
For project reason I have to measure the performance of OData service being accessed parallelly. And ...
- 流畅python学习笔记第十八章:使用asyncio包处理并发(二)
前面介绍了asyncio的用法.下面我们来看下如何用协程的方式来实现之前的旋转指针的方法 @asyncio.coroutine def spin(msg): write,flush=sys.stdou ...
随机推荐
- MySQL分布式数据库架构:分库、分表、排序、分页、分组、实现教程
MySQL分库分表总结: 单库单表 : 单库单表是最常见的数据库设计,例如,有一张用户(user)表放在数据库db中,所有的用户都可以在db库中的user表中查到. 单库多表 : 随着用户数量的增加, ...
- Ajax 的简介与使用
一.什么是Ajax Ajax 的全称是 Asynchronous JavaScript and XML(即异步的 JavaScript 和 XML),是一种在无需重新加载整个网页的情况下,能够更新部分 ...
- NodeJS开发博客(二) 接入数据库
1. mysql 数据库下载网址:https://dev.mysql.com/downloads/mysql/ 账号是 root 密码是 a1************ 网站账号是邮箱,密码是 Aa1* ...
- 混合应用 微信登录授权 微信登录认证失败 ios PGWXAPI错误-1 code:-100 / 安卓 message:invalid appsecret innerCode:40125
最近项目需要做微信登录,于是利用HTML5+ API Reference的OAuth模块管理客户端的用户登录授权验证功能,允许应用访问第三方平台的资源.(链接:https://www.dcloud.i ...
- 前端学习笔记--CSS3
本本记录了css3的样式:浏览器支持度.圆角边框.阴影.文字与文本.过渡.动画.2d旋转.3d旋转 浏览器支持度: 1.圆角边框 例:只要确定了x.y值,就能知道弧度 画一个圆形:长=宽,border ...
- POJ1185 炮兵阵地 和 POJ2411 Mondriaan's Dream
炮兵阵地 Language:Default 炮兵阵地 Time Limit: 2000MS Memory Limit: 65536K Total Submissions: 34008 Accepted ...
- 斜率优化DP总结
HDU3507 Print Article Zero has an old printer that doesn't work well sometimes. As it is antique, he ...
- 2.3 vue配置(上)
rm,在打包之前把上一次打包之后的东西删掉,然后webpack重新打包 通过DefinePlugin形成一个环境变量 HTML打包插件
- vue 传入一个对象的所有属性
- group_concat 排序并取前三个
substring_index(group_concat(distinct num order by num desc),',',3)