成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python 中的異步編程:Asyncio

開(kāi)發(fā) 后端
當(dāng)使用標(biāo)準(zhǔn)的 sleep方法時(shí),當(dāng)前線(xiàn)程會(huì)掛起等待。什么也不會(huì)做。實(shí)際上,標(biāo)準(zhǔn)的 sleep 過(guò)程中,當(dāng)前線(xiàn)程也會(huì)返回一個(gè) python 的解釋器,可以操作現(xiàn)有的其他線(xiàn)程,但這是另一個(gè)話(huà)題了。

[[190376]]

異步是怎么一回事?

在傳統(tǒng)的順序編程中, 所有發(fā)送給解釋器的指令會(huì)一條條被執(zhí)行。此類(lèi)代碼的輸出容易顯現(xiàn)和預(yù)測(cè)。 但是…

譬如說(shuō)你有一個(gè)腳本向3個(gè)不同服務(wù)器請(qǐng)求數(shù)據(jù)。 有時(shí),誰(shuí)知什么原因,發(fā)送給其中一個(gè)服務(wù)器的請(qǐng)求可能意外地執(zhí)行了很長(zhǎng)時(shí)間。想象一下從第二個(gè)服務(wù)器獲取數(shù)據(jù)用了10秒鐘。在你等待的時(shí)候,整個(gè)腳本實(shí)際上什么也沒(méi)干。如果你可以寫(xiě)一個(gè)腳本可以不去等待第二個(gè)請(qǐng)求而是僅僅跳過(guò)它,然后開(kāi)始執(zhí)行第三個(gè)請(qǐng)求,然后回到第二個(gè)請(qǐng)求,執(zhí)行之前離開(kāi)的位置會(huì)怎么樣呢。就是這樣。你通過(guò)切換任務(wù)最小化了空轉(zhuǎn)時(shí)間。盡管如此,當(dāng)你需要一個(gè)幾乎沒(méi)有I/O的簡(jiǎn)單腳本時(shí),你不想用異步代碼。

還有一件重要的事情要提,所有代碼在一個(gè)線(xiàn)程中運(yùn)行。所以如果你想讓程序的一部分在后臺(tái)執(zhí)行同時(shí)干一些其他事情,那是不可能的。

準(zhǔn)備開(kāi)始

這是 asyncio 主概念最基本的定義:

  • 協(xié)程— 消費(fèi)數(shù)據(jù)的生成器,但是不生成數(shù)據(jù)。Python 2.5 介紹了一種新的語(yǔ)法讓發(fā)送數(shù)據(jù)到生成器成為可能。我推薦查閱David Beazley “A Curious Course on Coroutines and Concurrency” 關(guān)于協(xié)程的詳細(xì)介紹。
  • 任務(wù)— 協(xié)程調(diào)度器。如果你觀(guān)察下面的代碼,你會(huì)發(fā)現(xiàn)它只是讓 event_loop 盡快調(diào)用它的_step ,同時(shí) _step 只是調(diào)用協(xié)程的下一步。
  1. class Task(futures.Future):   
  2.     def __init__(self, coro, loop=None): 
  3.         super().__init__(loop=loop) 
  4.         ... 
  5.         self._loop.call_soon(self._step) 
  6.  
  7.     def _step(self): 
  8.             ... 
  9.         try: 
  10.             ... 
  11.             result = next(self._coro) 
  12.         except StopIteration as exc: 
  13.             self.set_result(exc.value) 
  14.         except BaseException as exc: 
  15.             self.set_exception(exc) 
  16.             raise 
  17.         else
  18.             ... 
  19.             self._loop.call_soon(self._step) 
  • 事件循環(huán)— 把它想成 asyncio 的中心執(zhí)行器。

現(xiàn)在我們看一下所有這些如何融為一體。正如我之前提到的,異步代碼在一個(gè)線(xiàn)程中運(yùn)行。

從上圖可知:

1.消息循環(huán)是在線(xiàn)程中執(zhí)行

2.從隊(duì)列中取得任務(wù)

3.每個(gè)任務(wù)在協(xié)程中執(zhí)行下一步動(dòng)作

4.如果在一個(gè)協(xié)程中調(diào)用另一個(gè)協(xié)程(await <coroutine_name>),會(huì)觸發(fā)上下文切換,掛起當(dāng)前協(xié)程,并保存現(xiàn)場(chǎng)環(huán)境(變量,狀態(tài)),然后載入被調(diào)用協(xié)程

5.如果協(xié)程的執(zhí)行到阻塞部分(阻塞I/O,Sleep),當(dāng)前協(xié)程會(huì)掛起,并將控制權(quán)返回到線(xiàn)程的消息循環(huán)中,然后消息循環(huán)繼續(xù)從隊(duì)列中執(zhí)行下一個(gè)任務(wù)...以此類(lèi)推

6.隊(duì)列中的所有任務(wù)執(zhí)行完畢后,消息循環(huán)返回***個(gè)任務(wù)

異步和同步的代碼對(duì)比

現(xiàn)在我們實(shí)際驗(yàn)證異步模式的切實(shí)有效,我會(huì)比較兩段 python 腳本,這兩個(gè)腳本除了 sleep 方法外,其余部分完全相同。在***個(gè)腳本里,我會(huì)用標(biāo)準(zhǔn)的 time.sleep 方法,在第二個(gè)腳本里使用 asyncio.sleep 的異步方法。

這里使用 Sleep 是因?yàn)樗且粋€(gè)用來(lái)展示異步方法如何操作 I/O 的最簡(jiǎn)單辦法。

使用同步 sleep 方法的代碼:

  1. import asyncio   
  2. import time   
  3. from datetime import datetime 
  4.  
  5.  
  6. async def custom_sleep():   
  7.     print('SLEEP', datetime.now()) 
  8.     time.sleep(1) 
  9.  
  10. async def factorial(name, number):   
  11.     f = 1 
  12.     for i in range(2, number+1): 
  13.         print('Task {}: Compute factorial({})'.format(name, i)) 
  14.         await custom_sleep() 
  15.         f *= i 
  16.     print('Task {}: factorial({}) is {}\n'.format(name, number, f)) 
  17.  
  18.  
  19. start = time.time()   
  20. loop = asyncio.get_event_loop() 
  21.  
  22. tasks = [   
  23.     asyncio.ensure_future(factorial("A", 3)), 
  24.     asyncio.ensure_future(factorial("B", 4)), 
  25. loop.run_until_complete(asyncio.wait(tasks))   
  26. loop.close() 
  27.  
  28. end = time.time()   
  29. print("Total time: {}".format(end - start)) 

腳本輸出:

  1. Task A: Compute factorial(2)   
  2. SLEEP 2017-04-06 13:39:56.207479   
  3. Task A: Compute factorial(3)   
  4. SLEEP 2017-04-06 13:39:57.210128   
  5. Task A: factorial(3) is 6 
  6.  
  7. Task B: Compute factorial(2)   
  8. SLEEP 2017-04-06 13:39:58.210778   
  9. Task B: Compute factorial(3)   
  10. SLEEP 2017-04-06 13:39:59.212510   
  11. Task B: Compute factorial(4)   
  12. SLEEP 2017-04-06 13:40:00.217308   
  13. Task B: factorial(4) is 24 
  14.  
  15. Total time: 5.016386032104492 

使用異步 Sleep 的代碼:

  1. import asyncio   
  2. import time   
  3. from datetime import datetime 
  4.  
  5.  
  6. async def custom_sleep():   
  7.     print('SLEEP {}\n'.format(datetime.now())) 
  8.     await asyncio.sleep(1) 
  9.  
  10. async def factorial(name, number):   
  11.     f = 1 
  12.     for i in range(2, number+1): 
  13.         print('Task {}: Compute factorial({})'.format(name, i)) 
  14.         await custom_sleep() 
  15.         f *= i 
  16.     print('Task {}: factorial({}) is {}\n'.format(name, number, f)) 
  17.  
  18.  
  19. start = time.time()   
  20. loop = asyncio.get_event_loop() 
  21.  
  22. tasks = [   
  23.     asyncio.ensure_future(factorial("A", 3)), 
  24.     asyncio.ensure_future(factorial("B", 4)), 
  25. loop.run_until_complete(asyncio.wait(tasks))   
  26. loop.close() 
  27.  
  28. end = time.time()   
  29. print("Total time: {}".format(end - start)) 

腳本輸出:

  1. Task A: Compute factorial(2)   
  2. SLEEP 2017-04-06 13:44:40.648665 
  3.  
  4. Task B: Compute factorial(2)   
  5. SLEEP 2017-04-06 13:44:40.648859 
  6.  
  7. Task A: Compute factorial(3)   
  8. SLEEP 2017-04-06 13:44:41.649564 
  9.  
  10. Task B: Compute factorial(3)   
  11. SLEEP 2017-04-06 13:44:41.649943 
  12.  
  13. Task A: factorial(3) is 6 
  14.  
  15. Task B: Compute factorial(4)   
  16. SLEEP 2017-04-06 13:44:42.651755 
  17.  
  18. Task B: factorial(4) is 24 
  19.  
  20. Total time: 3.008226156234741 

從輸出可以看到,異步模式的代碼執(zhí)行速度快了大概兩秒。當(dāng)使用異步模式的時(shí)候(每次調(diào)用 await asyncio.sleep(1) ),進(jìn)程控制權(quán)會(huì)返回到主程序的消息循環(huán)里,并開(kāi)始運(yùn)行隊(duì)列的其他任務(wù)(任務(wù)A或者任務(wù)B)。

當(dāng)使用標(biāo)準(zhǔn)的 sleep方法時(shí),當(dāng)前線(xiàn)程會(huì)掛起等待。什么也不會(huì)做。實(shí)際上,標(biāo)準(zhǔn)的 sleep 過(guò)程中,當(dāng)前線(xiàn)程也會(huì)返回一個(gè) python 的解釋器,可以操作現(xiàn)有的其他線(xiàn)程,但這是另一個(gè)話(huà)題了。

推薦使用異步模式編程的幾個(gè)理由

很多公司的產(chǎn)品都廣泛的使用了異步模式,如 Facebook 旗下著名的 React Native 和 RocksDB 。像 Twitter 每天可以承載 50 億的用戶(hù)訪(fǎng)問(wèn),靠的也是異步模式編程。所以說(shuō),通過(guò)代碼重構(gòu),或者改變模式方法,就能讓系統(tǒng)工作的更快,為什么不去試一下呢?

責(zé)任編輯:武曉燕 來(lái)源: 36大數(shù)據(jù)
相關(guān)推薦

2017-08-02 15:00:12

PythonAsyncio異步編程

2011-02-22 08:49:16

.NET同步異步

2011-02-22 09:09:21

.NETAsync CTP異步

2020-02-21 08:00:00

Pythonasyncio編程語(yǔ)言

2023-07-14 15:10:17

PythonAsyncIO庫(kù)

2025-06-03 08:27:58

Python異步IO編程

2018-10-08 15:35:56

Python異步IO

2023-11-29 07:10:50

python協(xié)程異步編程

2011-02-24 12:53:51

.NET異步傳統(tǒng)

2024-12-23 08:00:45

2013-04-01 15:38:54

異步編程異步編程模型

2013-04-01 15:25:41

異步編程異步EMP

2023-08-30 08:43:42

asyncioaiohttp

2024-04-18 08:20:27

Java 8編程工具

2020-10-15 13:29:57

javascript

2021-12-10 07:47:30

Javascript異步編程

2021-08-02 11:13:28

人工智能機(jī)器學(xué)習(xí)技術(shù)

2023-08-01 08:43:29

Python多線(xiàn)程

2014-03-31 10:51:40

pythonasyncio

2015-04-22 10:50:18

JavascriptJavascript異
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 亚洲色欲色欲www | 久久久91| 成人欧美一区二区三区色青冈 | 日韩中文字幕视频在线观看 | 精品日韩电影 | 91精品国产91久久久久久 | 免费播放一级片 | 久久国产一区二区三区 | 国产一区二区在线免费观看 | 中文字幕一区二区三区在线观看 | 在线亚洲欧美 | 欧美黄色免费网站 | 国产精品高清一区二区三区 | 在线观看视频91 | 日韩精品一区二区三区视频播放 | 日韩视频在线观看一区二区 | 夜久久 | 免费黄色av网站 | 黄色毛片一级 | 国产免费观看一级国产 | 一本大道久久a久久精二百 国产成人免费在线 | 国产一二三区免费视频 | 午夜亚洲| 一本一道久久a久久精品综合蜜臀 | 欧美一区二区三区在线 | 国产精品激情 | 国内久久 | 国产色婷婷精品综合在线手机播放 | 五月天国产 | 亚洲精品久久久久久久久久久 | 国产欧美精品一区二区三区 | 91伊人网 | 日韩成年人视频在线 | 色欧美日韩 | av国产精品 | 毛片入口| 蜜桃五月天 | www.青娱乐 | 欧美黄色性生活视频 | 大学生a级毛片免费视频 | 国产精品成人一区二区 |