python 協(xié)程中的迭代器,生成器原理及應(yīng)用實(shí)例詳解
本文實(shí)例講述了python 協(xié)程中的迭代器,生成器原理及應(yīng)用。分享給大家供大家參考,具體如下:
1.迭代器理解
迭代器:
-
迭代器是訪(fǎng)問(wèn)可迭代對(duì)象的工具
-
迭代器是指用iter(obj)函數(shù)返回的對(duì)象(實(shí)例)
-
迭代器是指用next(it)函數(shù)獲取可迭代對(duì)象的數(shù)據(jù)
迭代器函數(shù)(iter和next)
-
iter(iterable)從可迭代對(duì)象中返回一個(gè)迭代器,iterable必須是能提供一個(gè)迭代器的對(duì)象
-
next(iterator) 從迭代器iterator中獲取下一了記錄,如果無(wú)法獲取下一條記錄,則觸發(fā)stoptrerator異常
說(shuō)明:
1.迭代器只能往前取值,不會(huì)后退
2.用iter函數(shù)可以返回一個(gè)可迭代對(duì)象的迭代器
2.迭代器的應(yīng)用
class Fabonacci(object): def __init__(self,all_num): self.all_num = all_num self.current_num = 0 self.a = 0 self.b = 1 def __iter__(self): return self def __next__(self): if self.current_num < self.all_num: ret = self.a self.a, self.b = self.b, self.a + self.b self.current_num += 1 return ret else: raise StopIteration fibo = Fabonacci(10) for num in fibo: print(num)
3.生成器的理解
生成器(generator)
-
是構(gòu)造新的可迭代對(duì)象的一種簡(jiǎn)單方式。一般的函數(shù)return只會(huì)返回單個(gè)值,而生成器并不是直接將可迭代值直接放入內(nèi)存中,而是以延遲的方式返回一個(gè)值序列,即每返回一個(gè)值之后暫停,直到下一個(gè)值被請(qǐng)求時(shí)再繼續(xù),可有效節(jié)省內(nèi)存占用。
-
要構(gòu)建一個(gè)生成器,則需要用到關(guān)鍵字yield,yield的作用與函數(shù)的返回值return有些類(lèi)似,通過(guò)在函數(shù)中將return替換成yield就是把函數(shù)變成生成器,帶有
yield 的函數(shù)不再是普通函數(shù),python
解釋器會(huì)將函數(shù)對(duì)象視為生成器對(duì)象,并且該生成器返回的是yield表達(dá)式生成的可迭代值序列,可通過(guò)for循環(huán)等方法依次讀取生成器返回的可迭代值序列 -
生成器生成的可迭代值只可以被讀取一次,每一次迭代都是按生成器代碼流程遇見(jiàn)yield表達(dá)式就返回值并記錄位置后中止留待下一次迭代,下一次迭代時(shí)執(zhí)行代碼的起始位置是從上一次記錄位置開(kāi)始,直至整個(gè)生成器代碼運(yùn)行結(jié)束。
4.生成器的應(yīng)用
1)
def create_num(all_num): a, b = 0, 1 current_num = 0 while current_num < all_num: # print(a) yield a a, b = b, a+b current_num += 1 obj = create_num(10) while True: try: ret = next(obj) print(ret) except Exception as ret: break
2)通過(guò)send啟動(dòng)生成器
send一般不會(huì)放到第一次啟動(dòng)生成器,如果非要這么做,那么傳遞None
錯(cuò)誤示范:
def create_num(all_num): a, b = 0, 1 current_num = 0 while current_num < all_num: ret = yield a print(">>>ret>>>", ret) a, b = b, a+b current_num += 1 obj = create_num(10) ret = obj.send("hello") #第一個(gè)就調(diào)用send方法 print(ret) ret = next(obj) print(ret)
正確示范:
def create_num(all_num): a, b = 0, 1 current_num = 0 while current_num < all_num: ret = yield a print(">>>ret>>>", ret) a, b = b, a+b current_num += 1 obj = create_num(10) ret = next(obj) print(ret) ret = obj.send("hello") print(ret)
def create_num(all_num): a, b = 0, 1 current_num = 0 while current_num < all_num: ret = yield a print(">>>ret>>>", ret) a, b = b, a+b current_num += 1 obj = create_num(10) ret = obj.send(None) print(ret) ret = next(obj) print(ret)
3).使用生成器完成多任務(wù)(并發(fā))
import time def task_1(): while True: print("----1----") time.sleep(0.1) yield def task_2(): while True: print("----2----") time.sleep(1) yield def main(): t1 = task_1() t2 = task_2() while True: next(t1) next(t2) if __name__ == '__main__': main()
4).gevent使用生成器
導(dǎo)入genvent庫(kù)
import gevent import time def f1(n): for i in range(n): print(gevent.getcurrent(),i) gevent.sleep(0.5) def f2(n): for i in range(n): print(gevent.getcurrent(),i) gevent.sleep(0.5) def f3(n): for i in range(n): print(gevent.getcurrent(),i) gevent.sleep(0.5) print("----1----") g1 = gevent.spawn(f1,5) print("----2----") g2 = gevent.spawn(f2,5) print("----3----") g3 = gevent.spawn(f3,5) g1.join() g2.join() g3.join()
修改time.sleep()成gevent.sleep()的簡(jiǎn)單方法:(打補(bǔ)丁)
只需要導(dǎo)入monkey,寫(xiě)一句代碼monkey.patch_all()
,運(yùn)行時(shí)就會(huì)自動(dòng)替換
import gevent import time from gevent import monkey monkey.patch_all() def f1(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) def f2(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) def f3(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) print("----1----") g1 = gevent.spawn(f1,5) print("----2----") g2 = gevent.spawn(f2,5) print("----3----") g3 = gevent.spawn(f3,5) g1.join() g2.join() g3.join()
創(chuàng)建多個(gè)gevent時(shí)不需一個(gè)一個(gè)添加join
import gevent import time from gevent import monkey monkey.patch_all() def f1(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) def f2(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) def f3(n): for i in range(n): print(gevent.getcurrent(),i) time.sleep(0.5) gevent.joinall([ gevent.spawn(f1,5), gevent.spawn(f2,5), gevent.spawn(f3,5)])
更多關(guān)于Python相關(guān)內(nèi)容可查看本站專(zhuān)題:《Python數(shù)據(jù)結(jié)構(gòu)與算法教程》、《Python Socket編程技巧總結(jié)》、《Python函數(shù)使用技巧總結(jié)》、《Python字符串操作技巧匯總》及《Python入門(mén)與進(jìn)階經(jīng)典教程》
希望本文所述對(duì)大家Python程序設(shè)計(jì)有所幫助。
相關(guān)文章
運(yùn)行tensorflow python程序,限制對(duì)GPU和CPU的占用操作
今天小編就為大家分享一篇運(yùn)行tensorflow python程序,限制對(duì)GPU和CPU的占用操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-02-02python性能測(cè)試對(duì)手機(jī)號(hào)綁定進(jìn)行壓測(cè)
這篇文章主要為大家介紹了python性能測(cè)試對(duì)手機(jī)號(hào)綁定進(jìn)行壓測(cè)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07使用python?pywin32模塊創(chuàng)建windows服務(wù)實(shí)例探究
這篇文章主要為大家介紹了使用python?pywin32模塊創(chuàng)建windows服務(wù)實(shí)現(xiàn)實(shí)例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2024-01-01Python交換字典鍵值對(duì)的四種方法實(shí)例
字典中有成對(duì)出現(xiàn)的鍵和值,但是字典中的鍵值對(duì)不是都能修改的,只有值才能修改,下面這篇文章主要給大家介紹了關(guān)于Python交換字典鍵值對(duì)的四種方法,需要的朋友可以參考下2022-12-12解決python升級(jí)引起的pip執(zhí)行錯(cuò)誤的問(wèn)題
今天小編就為大家分享一篇解決python升級(jí)引起的pip執(zhí)行錯(cuò)誤的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-06-06Python中pandas dataframe刪除一行或一列:drop函數(shù)詳解
今天小編就為大家分享一篇Python中pandas dataframe刪除一行或一列:drop函數(shù)詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07Django migrations 默認(rèn)目錄修改的方法教程
這篇文章主要介紹了Django migrations 默認(rèn)目錄修改的方法教程,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-09-09Python數(shù)據(jù)可視化庫(kù)seaborn的使用總結(jié)
這篇文章主要介紹了Python數(shù)據(jù)可視化庫(kù)seaborn的使用總結(jié),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2019-01-01