async 异步

什么是协程, 为什么要用协程

协程是用户态的线程,是实现多任务的一种方式,
为什么不用多线程实现多任务而是用协程呢? 因为协程的切换是在代码中逻辑切换的, 协程任务的切换不需要到cpu内核中进行切换, 因此,协程的切换要比线程快,资源消耗要比线程小,
在web开发中大都是io任务,python中的多线程有GIL锁,实际上也是单线程,无法真正利用cpu多核并行,况且, web 开发大都是 io堵塞等待, 多线程争抢到gil锁大多时间都耗费在io等待上
异步是基于事件轮询的多任务, 遇到io堵塞的时候会自动让出执行权给其他任务执行,这样的话就可以更好的压榨cpu, 提高性能

如何创建协程异步任务

很简单,只要在定义方法的前面加上async 那么这个方法就是异步协程异步任务了, 如:

async def async_task():
    print("这是一个协程异步任务")

t = async_task()

被async 装饰的任务就是一个异步任务, 异步任务之间调用的话他返回一个异步任务对象,并不会执行这个异步任务, 想要执行这个异步任务的话还需要 asycnio 包

asyncio

asyncio 是python3.4之后自带的一个异步任务模块, 无需额外下载, 异步任务的执行需要在事件循环(event_loop)中执行

事件循环 event_loop

事件循环是实现多任务的关键,你可以把它当做是一个 while True 循环
我们将异步任务放置到这个事件循环上,由他自己循环调用异步任务实现多任务, 异步任务在调用中如果遇到了 await 堵塞操作的时候会回自动的将这个任务挂起, 注册回调函数,挂起的时候事件循环就不会循环这个挂起任务
当注册的回调函数将这个异步任务唤醒的时候,这个任务才会重新放到事件循环中继续循环执行

异步任务的使用
import asyncio
import time


async def get_baidu():
    print("异步爬虫爬取百度, 时间是: ", time.time())
    await asyncio.sleep(2)  # 休息2秒,模拟io堵塞
    print("百度  异步爬虫结束, 时间是: ", time.time())


async def get_google():
    print("异步爬虫,爬取google 时间是: ", time.time())
    await asyncio.sleep(2)  # 休息2秒,模拟io堵塞
    print("google 异步爬虫结束   时间是: ", time.time())

task1 = get_baidu()
task2 = get_google()

# 获取事件循环
loop = asyncio.get_event_loop()

# 往事件循环中添加异步任务
loop.create_task(task1)
loop.create_task(task2)

# 开启事件循环
loop.run_forever()

注意, 在异步任务中 await 只能等待可等待的对象,普通的 time.sleep 没有实现可等待,所以要注意不要用错了

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值