日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

python 如何引入協程和原理分析

瀏覽:152日期:2022-07-03 17:13:08

相關概念

并發:指一個時間段內,有幾個程序在同一個cpu上運行,但是任意時刻只有一個程序在cpu上運行。比如說在一秒內cpu切換了100個進程,就可以認為cpu的并發是100。 并行:值任意時刻點上,有多個程序同時運行在cpu上,可以理解為多個cpu,每個cpu獨立運行自己程序,互不干擾。并行數量和cpu數量是一致的。

我們平時常說的高并發而不是高并行,是因為cpu的數量是有限的,不可以增加。

形象的理解:cpu對應一個人,程序對應喝茶,人要喝茶需要四個步驟(可以對應程序需要開啟四個線程):1燒水,2備茶葉,3洗茶杯,4泡茶。

并發方式:燒水的同時做好2備茶葉,3洗茶杯,等水燒好之后執行4泡茶。這樣比順序執行1234要省時間。

并行方式:叫來四個人(開啟四個進程),分別執行任務1234,整個程序執行時間取決于耗時最多的步驟。

同步 (注意同步和異步只是針對于I/O操作來講的)值調用IO操作時,必須等待IO操作完成后才開始新的的調用方式。 異步 指調用IO操作時,不必等待IO操作完成就開始新的的調用方式。 阻塞 指調用函數的時候,當前線程被掛起。 非阻塞 指調用函數的時候,當前線程不會被掛起,而是立即返回。

IO多路復用

sllect, poll, epoll都是IO多路復用的機制。IO多路復用就是通過這樣一種機制:一個進程可以監聽多個描述符,一旦某個描述符就緒(一般是讀就緒和寫就緒),能夠通知程序進行相應的操作。但select,poll,epoll本質上都是同步IO,因為他們都需要在讀寫事件就緒后自己負責進行讀寫(即將數據從內核空間拷貝到應用緩存)。也就是說這個讀寫過程是阻塞的。而異步IO則無需自己負責讀寫,異步IO的實現會負責把數據從內核拷貝到用戶空間。

select select函數監聽的文件描述符分三類:writefds、readfds、和exceptfds。調用后select函數會阻塞,直到描述符就緒(有數據可讀、寫、或者有except)或者超時(timeout指定等待時間,如果立即返回則設置為null),函數返回。當select函數返回后,可以通過遍歷fdset,來找到就緒的描述符。

優點:良好的跨平臺性(幾乎所有的平臺都支持)缺點:單個進程能夠監聽的文件描述符數量存在最大限制,在linux上一般為1024,可以通過修改宏定義甚至重新編譯內核來提升,但是這樣也會造成效率降低。

poll

  不同于select使用三個位圖來表示fdset的方式,poll使用的是pollfd的指針實現

pollfd結構包含了要監聽的event和發生的event,不再使用select“參數-值”傳遞的方式。同時pollfd并沒有最大數量限制(但是數量過大之后性能也是會下降)。和select函數一樣,poll返回后,需要輪詢pollfd來獲取就緒的描述符。

從上面看,select和poll都需要在返回后,通過遍歷文件描述符來獲取已經就緒的socket。事實上,同時連接的大量客戶端在同一時刻可能只有很少的處于就緒狀態,因此隨著監視的描述符數量的增長,其效率也會下降。

epoll

epoll是在linux2.6內核中國提出的,(windows不支持),是之前的select和poll增強版。相對于select和poll來說,epoll更加靈活,沒有描述符的限制。epoll使用一個文件描述符管理多個描述符,將用戶關系的文件描述符的時間存放到內核的一個時間表中。這樣在用戶控件和內核控件的coppy只需要一次。

如何選擇?

①在并發高同時連接活躍度不是很高的請看下,epoll比select好(網站或web系統中,用戶請求一個頁面后隨時可能會關閉)

②并發性不高,同時連接很活躍,select比epoll好。(比如說游戲中數據一但連接了就會一直活躍,不會中斷)

省略章節:由于在用到select的時候需要嵌套多層回調函數,然后印發一系列的問題,如可讀性差,共享狀態管理困難,出現異常排查復雜,于是引入協程,既操作簡單,速度又快。

協程

對于上面的問題,我們希望去解決這樣幾個問題:

采用同步的方式去編寫異步的代碼,使代碼的可讀性高,更簡便。 使用單線程去切換任務(就像單線程間函數之間的切換那樣,速度超快)

(1)線程是由操作系統切換的,單線程的切換意味著我們需要程序員自己去調度任務。

(2)不需要鎖,并發性高,如果單線程內切換函數,性能遠高于線程切換,并發性更高。

例如我們在做爬蟲的時候:

def get_url(url): html = get_html(url) # 此處網絡下載IO操作比較耗時,希望切換到另一個函數去執行 infos = parse_html(html)# 下載url中的htmldef get_html(url): pass# 解析網頁def parse_html(html): pass

意味著我們需要一個可以暫停的函數,對于此函數可以向暫停的地方穿入值。(回憶我們的生成器函數就可以滿足這兩個條件)所以就引入了協程。

生成器進階

生成器不僅可以產出值,還可以接收值,用send()方法。注意:在調用send()發送非None值之前必須先啟動生成器,可以用①next()②send(None)兩種方式激活

def gen_func(): html = yield ’http://www.baidu.com’ # yield 前面加=號就實現了1:可以產出值2:可以接受調用者傳過來的值 print(html) yield 2 yield 3 return ’bobby’if __name__ == ’__main__’: gen = gen_func() url = next(gen) print(url) html = ’bobby’ gen.send(html) # send方法既可以將值傳遞進生成器內部,又可以重新啟動生成器執行到下一yield位置。打印結果:http://www.baidu.combobby close()方法。

def gen_func(): yield ’http://www.baidu.com’ # yield 前面加=號就實現了1:可以產出值2:可以接受調用者傳過來的值 yield 2 yield 3 return ’bobby’if __name__ == ’__main__’: gen = gen_func() url = next(gen) gen.close() next(gen)輸出結果:StopIteration

特別注意:調用close.()之后, 生成器在往下運行的時候就會產生出一個GeneratorExit,單數如果用try捕獲異常的話,就算捕獲了遇到后面還有yield的話,還是不能往下運行了,因為一旦調用close方法生成器就終止運行了(如果還有next,就會會產生一個異常)所以我們不要去try捕捉該異常。(此注意可以先忽略)

def gen_func(): try: yield ’http://www.baidu.com’ except GeneratorExit: pass yield 2 yield 3 return ’bobby’if __name__ == ’__main__’: gen = gen_func() print(next(gen)) gen.close() next(gen)輸出結果:RuntimeError: generator ignored GeneratorExit 調用throw()方法。用于拋出一個異常。該異常可以捕捉忽略。

def gen_func(): yield ’http://www.baidu.com’ # yield 前面加=號就實現了1:可以產出值2:可以接受調用者傳過來的值 yield 2 yield 3 return ’bobby’if __name__ == ’__main__’: gen = gen_func() print(next(gen)) gen.throw(Exception, ’Download Error’)輸出結果: Download Error

yield from

先看一個函數:from itertools import chain

from itertools import chainmy_list = [1,2,3]my_dict = {’frank’:’yangchao’, ’ailsa’:’liuliu’}for value in chain(my_list, my_dict, range(5,10)): chain()方法可以傳入多個可迭代對象,然后分別遍歷之。 print(value)打印結果:123frankailsa56789

此函數可以用yield from 實現:yield from功能 1:從一個可迭代對象中將值逐個返回。

my_list = [1,2,3]my_dict = {’frank’:’yangchao’, ’ailsa’:’liuliu’}def chain(*args, **kwargs): for itemrable in args: yield from itemrablefor value in chain(my_list, my_dict, range(5,10)): print(value)

看如下代碼:

def gen(): yield 1def g1(gen): yield from gendef main(): g = g1(gen) g.send(None)

代碼分析:此代碼中main調用了g1, main就叫作調用方, g1叫做委托方, gen 叫做子生成器yield from將會在調用方main與子生成器gen之間建立一個雙向通道。(意味著可以直接越過委托方)

例子:當委托方middle()中使用yield from 的時候,調用方main直接和子生成器sales_sum形成數據通道。

final_result = {}def sales_sum(pro_name): total = 0 nums = [] while True: x = yield print(pro_name+’銷量’, x) if not x: break total += x nums.append(x) return total, nums #程序運行到return的時候,會將return的返回值返回給委托方,即middle中的final_result[key]def middle(key): while True: #相當于不停監聽sales_sum是否有返回數據,(本例中有三次返回) final_result[key] = yield from sales_sum(key) print(key +’銷量統計完成!!’)def main(): data_sets = { ’面膜’:[1200, 1500, 3000], ’手機’:[88, 100, 98, 108], ’衣服’:[280, 560,778,70], } for key, data_set in data_sets.items(): print(’start key’, key) m = middle(key) m.send(None) # 預激生成器 for value in data_set: m.send(value) m.send(None)# 發送一個None使sales_sum中的x值為None退出while循環 print(final_result)if __name__ == ’__main__’: main()結果:start key 面膜面膜銷量 1200面膜銷量 1500面膜銷量 3000面膜銷量 None面膜銷量統計完成!!start key 手機手機銷量 88手機銷量 100手機銷量 98手機銷量 108手機銷量 None手機銷量統計完成!!start key 衣服衣服銷量 280衣服銷量 560衣服銷量 778衣服銷量 70衣服銷量 None衣服銷量統計完成!!{’面膜’: (5700, [1200, 1500, 3000]), ’手機’: (394, [88, 100, 98, 108]), ’衣服’: (1688, [280, 560, 778, 70])}

也許有人會好奇,為什么不能直接用main()函數直接去調用sales_sum呢?加一個委托方使代碼復雜化了。看以下直接用main()函數直接去調用sales_sum代碼:

def sales_sum(pro_name): total = 0 nums = [] while True: x = yield print(pro_name+’銷量’, x) if not x: break total += 1 nums.append(x) return total, numsif __name__ == ’__main__’: my_gen = sales_sum(’面膜’) my_gen.send(None) my_gen.send(1200) my_gen.send(1500) my_gen.send(3000) my_gen.send(None)輸出結果:面膜銷量 1200面膜銷量 1500面膜銷量 3000面膜銷量 NoneTraceback (most recent call last): File 'D:/MyCode/Cuiqingcai/Flask/test01.py', line 56, in <module> my_gen.send(None)StopIteration: (3, [1200, 1500, 3000])

從上述代碼可以看出,即使數據return結果出來了,還是會返回一個exception,由此可以看出yield from的一個最大優點就是當子生成器運行時候出現異常,yield from可以直接自動處理這些異常。

yield from 功能總結:

子生成器生產的值,都是直接給調用方;調用發通過.send()發送的值都是直接傳遞給子生成器,如果傳遞None,會調用子生成器的next()方法,如果不是None,會調用子生成器的sen()方法。子生成器退出的時候,最后的return EXPR,會觸發一個StopIteration(EXPR)異常yield from 表達式的值,是子生成器終止時,傳遞給StopIteration異常的第一個參數。如果調用的時候出現了StopIteration異常,委托方生成器恢復運行,同時其他的異常向上冒泡。傳入委托生成器的異常里,除了GeneratorExit之后,其他所有異常全部傳遞給子生成器的.throw()方法;如果調用.throw()的時候出現StopIteration異常,那么就恢復委托生成器的運行,其他的異常全部向上冒泡如果在委托生成器上調用.close()或傳入GeneratorExit異常,會調用子生成器的.close()方法,沒有就不調用,如果在調用.close()時候拋出了異常,那么就向上冒泡,否則的話委托生成器跑出GeneratorExit 異常。

以上就是python 如何引入協程和原理分析的詳細內容,更多關于python 協程的資料請關注好吧啦網其它相關文章!

標簽: Python 編程
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
亚洲精品在线二区| 亚洲精品午夜av福利久久蜜桃| 久久xxxx| 丝袜脚交一区二区| 亚洲一区二区小说| 亚洲精品九九| 久久xxxx| 日韩高清中文字幕一区| 日韩av在线免费观看不卡| 日韩精品一区二区三区中文| 亚洲影视一区| 国产亚洲久久| 老司机精品视频网| 国产精品yjizz视频网| 亚洲不卡av不卡一区二区| 婷婷成人在线| 蜜桃视频在线观看一区二区| 日韩精品三级| 久久久免费人体| 97精品97| 日韩视频一区二区三区在线播放免费观看 | 99久久精品网站| 一区二区自拍| 日本精品另类| 成人污污视频| 国产在线不卡| 中文字幕一区二区av| 欧美日韩一区二区高清| 国际精品欧美精品| 国产在线欧美| 日本成人中文字幕| 精品美女久久| 影音国产精品| 男人的天堂久久精品| 久久精品99国产精品| 日韩欧美一区免费| 六月婷婷一区| 美女久久精品| 99久久婷婷| 亚洲精品免费观看| 高清一区二区| 美女网站久久| 免费亚洲婷婷| 亚洲综合精品| 国产精品2023| 蜜臀91精品国产高清在线观看| 日韩福利视频一区| 日韩免费在线| 亚洲无线观看| 在线人成日本视频| 一区二区三区四区精品视频| 风间由美中文字幕在线看视频国产欧美| 激情欧美国产欧美| 国产乱人伦丫前精品视频| 999久久久91| 18国产精品| 黄色免费成人| 国产一区二区三区日韩精品 | 久久精品卡一| 日韩av一区二区三区四区| 91精品婷婷色在线观看| 三级久久三级久久久| 在线人成日本视频| 91麻豆精品| 日韩午夜电影| 欧美www视频在线观看| 亚洲欧美日本国产| 91精品综合| 精品亚洲二区| 亚洲三级精品| 久久免费国产| 欧美xxxx性| 亚洲免费资源| 美女网站一区| 精品午夜av| 日本成人在线一区| 午夜在线精品偷拍| 高清av不卡| 国产精品第一| 日韩欧美高清一区二区三区| 在线视频观看日韩| 国产66精品| 国产精品一区二区中文字幕| 日韩精品一级二级| 国产在线不卡| 天堂√8在线中文| 欧美精品二区| 久久国产乱子精品免费女| 蘑菇福利视频一区播放| 国产麻豆久久| 久久久精品国产**网站| 午夜亚洲福利| 亚洲免费播放| 亚洲性色av| 国产传媒在线观看| 国产欧美三级| 视频国产精品| 免费视频一区二区| 亚洲精品va| 九九久久婷婷| 四虎884aa成人精品最新| 国产精品成人a在线观看| 国产精品久久久久久久久久久久久久久 | 群体交乱之放荡娇妻一区二区| 国产精品成人自拍| 日本欧美在线| 综合国产精品| 免费不卡在线观看| 9久re热视频在线精品| 蜜臀av免费一区二区三区| 欧美三区四区| 日韩精品一区二区三区免费观影| 精品一区视频| 久久字幕精品一区| 精品五月天堂| 精品一区91| 精品高清久久| 国产一区二区三区四区| 日韩1区2区| 国产精品专区免费| 亚洲成人国产| 最新亚洲激情| 免费在线观看成人| 婷婷视频一区二区三区| 综合国产在线| 91国内精品| 欧美国产不卡| 麻豆精品久久| 美女视频免费精品| 福利片在线一区二区| 国产福利片在线观看| 波多视频一区| 伊人久久高清| 欧美/亚洲一区| 欧美精品自拍| 免费日本视频一区| 91大神在线观看线路一区| 国产欧美日韩| 久久精品午夜| 国产99在线| 在线日韩中文| 麻豆久久精品| 亚洲日产国产精品| 欧美日本不卡高清| 国产一区二区三区不卡视频网站| 韩日一区二区| 国产中文一区| 亚洲精品一级二级三级| 国产精品成人3p一区二区三区| 视频在线不卡免费观看| 欧美日韩在线播放视频| 久久xxxx| 久久狠狠亚洲综合| 国产精品久久久久蜜臀| 欧美三级网址| 一区免费视频| 国产日韩欧美中文在线| 成人一区而且| 伊人久久大香线蕉av超碰演员| 日韩二区在线观看| 97精品视频在线看| 快she精品国产999| 日韩大片在线播放| 蜜桃久久久久久久| 国产日韩电影| 亚洲一卡久久| 欧洲精品一区二区三区| 伊人久久婷婷| 久久国产三级精品| www在线观看黄色| 国产精品嫩草99av在线| 欧美日韩亚洲一区| 久久精品国产久精国产爱| 日韩免费福利视频| 亚洲一级淫片| 精品一区二区三区四区五区| 欧美粗暴jizz性欧美20| 日本不卡视频在线| 最近高清中文在线字幕在线观看1| 韩日一区二区三区| 欧美精品中文字幕亚洲专区| 日韩美女一区二区三区在线观看| 免费人成网站在线观看欧美高清| 国产精品激情| 一本一道久久a久久精品蜜桃| 欧美一区二区三区免费看| 日韩欧美少妇| 日韩激情av在线| 日韩精品免费一区二区三区| 日韩精品免费观看视频| 欧美三区四区| 日韩高清在线不卡| 日本免费一区二区三区四区| 日本一区二区三区中文字幕| 欧美国产小视频| 日韩一区欧美二区| 日韩毛片视频| 国产精品亚洲综合久久| 99成人在线| 国产精品xx|