Python函數(shù)生成器原理及使用詳解
1.python函數(shù)運(yùn)行原理
import inspect frame = None def foo(): bar() def bar(): global frame frame = inspect.currentframe() pass # python解釋器 python.exe 會用一個叫做PyEval_EvalFrameEx(c語言函數(shù))去執(zhí)行foo函數(shù),首先會創(chuàng)建一個棧幀(stack frame), """ python在運(yùn)行前會編譯成字節(jié)碼對象 當(dāng)foo調(diào)用bar函數(shù)進(jìn),又會創(chuàng)建一個棧幀, 關(guān)鍵是所有的棧幀都是分配在堆內(nèi)存, 堆內(nèi)存有個特點(diǎn),不手動釋放,就會一直存在 這就決定了棧幀可以獨(dú)立于調(diào)用者存在. """ # import dis # print(dis.dis(foo)) # 查看foo函數(shù)的字節(jié)碼 foo() #先調(diào)用一下foo函數(shù) ,這個frame就有值. print(frame.f_code.co_name) # bar 查看這個棧幀, bar 所以還是可以拿到bar的棧幀,然后就可以調(diào)用bar函數(shù) caller_frame = frame.f_back # 當(dāng)前frame棧幀的調(diào)用者的棧幀 print(caller_frame.f_code.co_name) # foo , 也可以拿到bar函數(shù)的棧幀
python中函數(shù)的調(diào)用就是創(chuàng)建棧幀的過程,而這些創(chuàng)建的棧幀都是存放在堆上面,不釋放就永久存在,所以我們拿到每個函數(shù)對應(yīng)的棧幀,就可以調(diào)用這個函數(shù).
java就不行了,函數(shù)執(zhí)行完畢,直接彈棧完蛋.
2.生成器執(zhí)行原理
測試代碼
def gen_fun(): yield 1 name = 'admin' yield 2 gender = 'male' return 3
看看測試代碼對應(yīng)的字節(jié)碼文件
0 LOAD_CONST 1 (1) YIELD_VALUE POP_TOP 6 LOAD_CONST 2 ('admin') STORE_FAST 0 (name) 10 LOAD_CONST 3 (2) YIELD_VALUE POP_TOP 16 LOAD_CONST 4 ('male') STORE_FAST 1 (gender) 20 LOAD_CONST 5 (3) RETURN_VALUE None
測試gi_frame
# 在沒有執(zhí)行生成器時 print(gen.gi_frame.f_lasti) # -1 ,在沒有調(diào)用next方法迭代時,f_lasti 等于-1, 表示還沒開始呢 print(gen.gi_frame.f_locals) # {} # 執(zhí)行第一行 next(gen) print(gen.gi_frame.f_lasti) # 2 # 執(zhí)行一行next后,代碼停在了第二行,看上面字節(jié)碼文件 print(gen.gi_frame.f_locals) # {} # 再執(zhí)行一次 next(gen) print(gen.gi_frame.f_lasti) # 12 # 又執(zhí)行一次next之后,程序停在了12行 print(gen.gi_frame.f_locals) # {'name': 'admin'}
由上面的測試代碼可以知道,在生成器的gi_frame對象中維護(hù)著兩個重要的屬性f_lasti和f_locals.
f_lasti記錄著當(dāng)前代碼運(yùn)行到哪一行了(注意這里的那一行是指編譯之后的字節(jié)碼文件)
f_locals維護(hù)著當(dāng)前生成器中的屬性字段
有了這兩個屬性,生成器就知道下一次next從哪兒開始執(zhí)行了....
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
淺談Python3.10 和 Python3.9 之間的差異
多年來,Python 進(jìn)行了大量升級,并且在新版本中添加了許多功能。本文就詳細(xì)的介紹 一下Python3.10 和 Python3.9差異,感興趣的朋友可以了解一下2021-09-09Python函數(shù)的返回值、匿名函數(shù)lambda、filter函數(shù)、map函數(shù)、reduce函數(shù)用法實(shí)例分析
這篇文章主要介紹了Python函數(shù)的返回值、匿名函數(shù)lambda、filter函數(shù)、map函數(shù)、reduce函數(shù)用法,結(jié)合實(shí)例形式分析了Python函數(shù)的返回值、匿名函數(shù)lambda、filter函數(shù)、map函數(shù)、reduce函數(shù)相關(guān)功能、原理與使用技巧,需要的朋友可以參考下2019-12-12Python基于twisted實(shí)現(xiàn)簡單的web服務(wù)器
這篇文章主要介紹了Python基于twisted實(shí)現(xiàn)簡單的web服務(wù)器,可模擬出簡單的web服務(wù)器功能,是很實(shí)用的技巧,需要的朋友可以參考下2014-09-09Python實(shí)現(xiàn)文件信息進(jìn)行合并實(shí)例代碼
這篇文章主要介紹了Python實(shí)現(xiàn)文件信息進(jìn)行合并實(shí)例代碼,具有一定借鑒價值,需要的朋友可以參考下2018-01-01Python中read,readline和readlines的區(qū)別案例詳解
這篇文章主要介紹了Python中read,readline和readlines的區(qū)別案例詳解,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下2021-09-09Django用戶認(rèn)證系統(tǒng) Web請求中的認(rèn)證解析
這篇文章主要介紹了Django用戶認(rèn)證系統(tǒng) Web請求中的認(rèn)證解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-08-08python中pivot()函數(shù)基礎(chǔ)知識點(diǎn)
在本篇內(nèi)容里小編給大家分享的是一篇關(guān)于python中pivot()函數(shù)基礎(chǔ)知識點(diǎn)內(nèi)容,對此有興趣的朋友們可以參考學(xué)習(xí)下。2021-01-01pytorch DataLoader的num_workers參數(shù)與設(shè)置大小詳解
這篇文章主要介紹了pytorch DataLoader的num_workers參數(shù)與設(shè)置大小詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-05-05