天天看點

淺談猴子更新檔

淺談猴子更新檔

屬性在運作時的動态替換,叫做猴子更新檔(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、在使用猴子更新檔的時候,還應注意如下事項: 

  • 基本上隻追加功能