
屬性在運作時的動态替換,叫做猴子更新檔(Monkey Patch)。
為什麼叫猴子更新檔?
屬性的運作時替換和猴子也沒什麼關系,關于猴子更新檔的由來網上查到兩種說法:
1、這個詞原來為Guerrilla Patch,雜牌軍、遊擊隊,說明這部分不是原裝的,在英文裡guerilla發音和gorllia(猩猩)相似,再後來就寫了monkey(猴子)。
2、還有一種解釋是說由于這種方式将原來的代碼弄亂了(messing with it),在英文裡叫monkeying about(頑皮的),是以叫做Monkey Patch。
猴子更新檔的叫法有些莫名其妙,隻要和“子產品運作時替換的功能”對應就行了。
猴子更新檔的用法
1、運作時動态替換子產品的方法
stackoverflow上有兩個比較熱的例子:
consider a class that has a method get_data. This method does an external lookup (on a database or web API, for example), and various other methods in the class call it. However, in a unit test, you don't want to depend on the external data source - so you dynamically replace the get_data method with a stub that returns some fixed data.
假設一個類有一個方法get_data。這個方法做一些外部查詢(如查詢資料庫或者Web API等),類裡面的很多其他方法都調用了它。
然而,在一個單元測試中,你不想依賴外部資料源。是以你用啞方法态替換了這個get_data方法,啞方法隻傳回一些測試資料。
另一個例子引用了,Zope wiki上對Monkey Patch解釋:
from SomeOtherProduct.SomeModule import SomeClass def speak(self): return "ook ook eee eee eee!" SomeClass.speak = speak
還有一個比較實用的例子,很多代碼用到 import json,後來發現ujson性能更高,如果覺得把每個檔案的import json 改成 import ujson as json成本較高,或者說想測試一下用ujson替換json是否符合預期,隻需要在入口加上:
import json import ujson def monkey_patch_json(): json.__name__ = 'ujson' json.dumps = ujson.dumps json.loads = ujson.loads monkey_patch_json()
2、運作時動态增加子產品的方法
這種場景也比較多,比如我們引用團隊通用庫裡的一個子產品,又想豐富子產品的功能,除了繼承之外也可以考慮用Monkey Patch。
3、在使用猴子更新檔的時候,還應注意如下事項:
- 基本上隻追加功能