python 執(zhí)行函數(shù)的九種方法
方法一:直接調(diào)用函數(shù)運(yùn)行
這種是最簡(jiǎn)單且直觀的方法
def task(): print("running task") task()
如果是在類中,也是如此
class Task: def task(self): print("running task") Task().task()
方法二:使用偏函數(shù)來(lái)執(zhí)行
在 functools 這個(gè)內(nèi)置庫(kù)中,有一個(gè) partial 方法專門用來(lái)生成偏函數(shù)。
def power(x, n): s = 1 while n > 0: n = n - 1 s = s * x return s from functools import partial power_2=partial(power, n=2) power_2(2) # output: 4 power_2(3) # output: 9
方法三:使用 eval 動(dòng)態(tài)執(zhí)行
如果你有需要?jiǎng)討B(tài)執(zhí)行函數(shù)的需要,可以使用 eval + 字符串 來(lái)執(zhí)行函數(shù)。
import sys def pre_task(): print("running pre_task") def task(): print("running task") def post_task(): print("running post_task") argvs = sys.argv[1:] for action in argvs: eval(action)()
運(yùn)行效果如下
$ python demo.py pre_task task post_task running pre_task running task running post_task
方法四:使用 getattr 動(dòng)態(tài)獲取執(zhí)行
若把所有的函數(shù)是放在類中,并定義成靜態(tài)方法,那就不需要用 eval 了,接著使用 getattr 去獲取并調(diào)用。
import sys class Task: @staticmethod def pre_task(): print("running pre_task") @staticmethod def task(): print("running task") @staticmethod def post_task(): print("running post_task") argvs = sys.argv[1:] task = Task() for action in argvs: func = getattr(task, action) func()
方法五:使用類本身的字典
我們都知道對(duì)象都有一個(gè) __dict__() 的魔法方法,存放所有對(duì)象的屬性及方法。
到這里,大家可以思考一下, 如果還是上面的代碼,我直接取實(shí)例的 __dict__() 能不能取到函數(shù)呢?
我相信很多人都會(huì)答錯(cuò)。
上面我們定義的是靜態(tài)方法,靜態(tài)方法并沒(méi)有與實(shí)例進(jìn)行綁定,因此靜態(tài)方法是屬于類的,但是不是屬于實(shí)例的,實(shí)例雖然有使用權(quán)(可以調(diào)用),但是并沒(méi)有擁有權(quán)。
因此要想通過(guò) __dict__ 獲取函數(shù),得通過(guò)類本身 Task,取出來(lái)的函數(shù),調(diào)用方法和平時(shí)的也不一樣,必須先用 __func__ 獲取才能調(diào)用。
import sys class Task: @staticmethod def pre_task(): print("running pre_task") func = Task.__dict__.get("pre_task") func.__func__()
方法六:使用 global() 獲取執(zhí)行
上面放入類中,只是為了方便使用 getattr 的方法,其實(shí)不放入類中,也是可以的。此時(shí)你需要借助 globals() 或者 locals() ,它們本質(zhì)上就是一個(gè)字典,你可以直接 get 來(lái)獲得函數(shù)。
import sys def pre_task(): print("running pre_task") def task(): print("running task") def post_task(): print("running post_task") argvs = sys.argv[1:] for action in argvs: globals().get(action)()
方法七:從文本中編譯運(yùn)行
先定義一個(gè)字符串,內(nèi)容是你函數(shù)的內(nèi)容,比如上面的 pre_task ,再通過(guò) compile 函數(shù)編進(jìn) 編譯,轉(zhuǎn)化為字節(jié)代碼,最后再使用 exec 去執(zhí)行它。
pre_task = """ print("running pre_task") """ exec(compile(pre_task, '<string>', 'exec'))
若你的代碼是放在一個(gè) txt 文本中,雖然無(wú)法直接導(dǎo)入運(yùn)行,但仍然可以通過(guò) open 來(lái)讀取,最后使用 compile 函數(shù)編譯運(yùn)行。
with open('source.txt') as f: source = f.read() exec(compile(source, 'source.txt', 'exec'))
方法八:使用 attrgetter 獲取執(zhí)行
在 operator 這個(gè)內(nèi)置庫(kù)中,有一個(gè)獲取屬性的方法,叫 attrgetter ,獲取到函數(shù)后再執(zhí)行。
from operator import attrgetter class People: def speak(self, dest): print("Hello, %s" %dest) p = People() caller = attrgetter("speak") caller(p)("明哥")
方法九:使用 methodcaller 執(zhí)行
同樣還是 operator 這個(gè)內(nèi)置庫(kù),有一個(gè) methodcaller 方法,使用它,也可以做到動(dòng)態(tài)調(diào)用實(shí)例方法的效果。
from operator import methodcaller class People: def speak(self, dest): print("Hello, %s" %dest) caller = methodcaller("speak", "明哥") p = People() caller(p)
以上就是我總結(jié)的函數(shù)執(zhí)行的十種方法,很多方法,大家也都知道,但是也有幾個(gè)方法,幾乎是見(jiàn)不到的,尤其是后面使用 operator 庫(kù)的那兩種方法。
以上就是python 執(zhí)行函數(shù)的九種方法的詳細(xì)內(nèi)容,更多關(guān)于python 執(zhí)行函數(shù)的方法的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
tensorflow2.0的函數(shù)簽名與圖結(jié)構(gòu)(推薦)
這篇文章主要介紹了tensorflow2.0的函數(shù)簽名與圖結(jié)構(gòu),本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-04-04關(guān)于python的bottle框架跨域請(qǐng)求報(bào)錯(cuò)問(wèn)題的處理方法
這篇文章主要介紹了關(guān)于python的bottle框架跨域請(qǐng)求報(bào)錯(cuò)問(wèn)題的處理方法,需要的朋友可以參考下2017-03-03Python?中設(shè)置請(qǐng)求的最大重試次數(shù)示例代碼
本篇文章介紹了為什么我們會(huì)收到錯(cuò)誤消息,指出超出了最大重試次數(shù),以及我們?nèi)绾卧?Python?中為請(qǐng)求設(shè)置?max_retries,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),需要的朋友參考下吧2023-06-06python將三維數(shù)組展開成二維數(shù)組的實(shí)現(xiàn)
今天小編就為大家分享一篇python將三維數(shù)組展開成二維數(shù)組的實(shí)現(xiàn),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2019-11-11Django 限制用戶訪問(wèn)頻率的中間件的實(shí)現(xiàn)
這篇文章主要介紹了Django 限制用戶訪問(wèn)頻率的中間件的實(shí)現(xiàn),小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-08-08