新聞中心
思路一:簡單地替換
當(dāng)你想對類實例的方法進(jìn)行替換時,你可能想到的是直接對他進(jìn)行粗暴地替換:

公司主營業(yè)務(wù):成都網(wǎng)站制作、網(wǎng)站設(shè)計、移動網(wǎng)站開發(fā)等業(yè)務(wù)。幫助企業(yè)客戶真正實現(xiàn)互聯(lián)網(wǎng)宣傳,提高企業(yè)的競爭能力。創(chuàng)新互聯(lián)公司是一支青春激揚、勤奮敬業(yè)、活力青春激揚、勤奮敬業(yè)、活力澎湃、和諧高效的團(tuán)隊。公司秉承以“開放、自由、嚴(yán)謹(jǐn)、自律”為核心的企業(yè)文化,感謝他們對我們的高要求,感謝他們從不同領(lǐng)域給我們帶來的挑戰(zhàn),讓我們激情的團(tuán)隊有機會用頭腦與智慧不斷的給客戶帶來驚喜。創(chuàng)新互聯(lián)公司推出孟州免費做網(wǎng)站回饋大家。
- class People:
- def speak(self):
- print("hello, world")
- def speak(self):
- print("hello, python")
- p = People()
- p.speak = speak
- p.speak()
但當(dāng)你試著執(zhí)行這段代碼的時候,就會發(fā)現(xiàn)行不通,它提示我們要傳入 self 參數(shù):
- Traceback (most recent call last):
- File "/Users/MING/Code/Python/demo.py", line 12, in
- p.speak()
- TypeError: speak() missing 1 required positional argument: 'self'
不對啊~ self 不是實例本身嗎?函數(shù)不是一直就這么寫的?
實際上你這么替換,speak 就變成了一個 function,而不是一個和實例綁定的 method ,你可以把替換前后的 speak 打印出來
- p = People()
- print(p.speak)
- p.speak = speak
- print(p.speak)
輸出結(jié)果如下,區(qū)別非常明顯
>
這種方法,只能用在替換不與實例綁定的靜態(tài)方法上,不然你每次調(diào)用的時候,就得手動傳入實例本身,但這樣調(diào)用就會變得非常怪異。
思路二:利用 im_func
有 Python 2 使用經(jīng)驗的朋友,可以會知道類實例的方法,都有 im_func 和 im_class 屬性,分別指向了該方法的函數(shù)和類。
很抱歉的是,這些在 Python3 中全都取消了,意味你無法再使用 im_func 和 im_class 。
但即使你身處 Python 2 的環(huán)境下,你想通過 im_func 去直接替換函數(shù),也仍然是有問題的。
因為在 Python2 中不推薦普通用戶對類實例的方法進(jìn)行替換,所以 Python 給類實例的方法賦予了只讀屬性
思路三:非常危險的字節(jié)碼替換
表層不行,但這個方法在字節(jié)碼層面卻是可行的
這種方法,非常的粗暴且危險,他會直接影響到使用 People 的所有實例的 speak 方法,因此這種方法千萬不要使用。
思路四:利用 types 綁定方法
在 types 中有一個 MethodType,可以將普通方法與實例進(jìn)行綁定。
綁定后,就可以直接替換掉原實例的 speak 方法了,完整代碼如下:
- import types
- class People:
- def speak(self):
- print("hello, world")
- def speak(self):
- print("hello, python")
- p = People()
- p.speak = types.MethodType(speak, p)
- p.speak()
這種方法,最為安全,不會影響其他實例。并且 Python 2 和 Python 3 都適用,是官方推薦的一種做法。
總結(jié)一下
- 直接替換:只適用于靜態(tài)方法
- 使用 im_func 替換:行不通
- 使用 im_func.func_code 替換字節(jié)碼:非常危險,請不要使用
- 使用 types.MethodType 進(jìn)行方法綁定:安全且有效,推薦使用
新聞名稱:替換一個實例方法,沒你想的那么簡單
文章出自:http://www.dlmjj.cn/article/dhgpejg.html


咨詢
建站咨詢
