Python函數(shù)生成器原理及使用詳解
1.python函數(shù)運行原理
import inspectframe = Nonedef 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在運行前會編譯成字節(jié)碼對象當foo調(diào)用bar函數(shù)進,又會創(chuàng)建一個棧幀,關(guān)鍵是所有的棧幀都是分配在堆內(nèi)存, 堆內(nèi)存有個特點,不手動釋放,就會一直存在這就決定了棧幀可以獨立于調(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 # 當前frame棧幀的調(diào)用者的棧幀print(caller_frame.f_code.co_name) # foo , 也可以拿到bar函數(shù)的棧幀
python中函數(shù)的調(diào)用就是創(chuàng)建棧幀的過程,而這些創(chuàng)建的棧幀都是存放在堆上面,不釋放就永久存在,所以我們拿到每個函數(shù)對應的棧幀,就可以調(diào)用這個函數(shù).
java就不行了,函數(shù)執(zhí)行完畢,直接彈棧完蛋.
2.生成器執(zhí)行原理
測試代碼
def gen_fun(): yield 1 name = ’admin’ yield 2 gender = ’male’ return 3
看看測試代碼對應的字節(jié)碼文件
0 LOAD_CONST1 (1)YIELD_VALUEPOP_TOP 6 LOAD_CONST2 (’admin’)STORE_FAST0 (name) 10 LOAD_CONST3 (2)YIELD_VALUEPOP_TOP 16 LOAD_CONST4 (’male’)STORE_FAST1 (gender) 20 LOAD_CONST5 (3)RETURN_VALUENone
測試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對象中維護著兩個重要的屬性f_lasti和f_locals.
f_lasti記錄著當前代碼運行到哪一行了(注意這里的那一行是指編譯之后的字節(jié)碼文件)
f_locals維護著當前生成器中的屬性字段
有了這兩個屬性,生成器就知道下一次next從哪兒開始執(zhí)行了....
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網(wǎng)。
相關(guān)文章:
