猴子补丁
根据维基百科上的解释,猴子补丁是一种让程序行为在运行时拓展或变更的方法
定义
猴子补丁的定义取决于使用它的社区,在Ruby、Python等动态编程语言中,猴子补丁仅指在运行时动态改变类或模块,为的是将第三方代码打补丁在不按预期运行的bug或者feature上
例子
在运行时改变math.pi
的值
>>> import math
>>> math.pi
3.141592653589793
>>> math.pi = 3
>>> math.pi
3
>>> ================================ RESTART ================================
>>> import math
>>> math.pi
3.141592653589793
>>>
用途
猴子补丁在单元测试的时候有很好的用处,假设我们在使用requests时:
import requests
def get_page(url):
try:
s = requests.get(url)
except requests.HTTPError as e:
print("HTTPError")
我们想要测试get_page是否正确处理了requests.HTTPError
,而通常requests.get
不太会引发这个异常,因此我们可以通过猴子补丁来进行测试:
import requests
def get_page(url):
try:
s = requests.get(url)
except requests.HTTPError as e:
print("HTTPError")
def get(url):
raise requests.HTTPError
if __name__ == "__main__":
requests.get = get # 猴子补丁
get_page("123")
这里只是为了写个例子,并没有按照通常的单元测试写法,在主函数的第一行,我们为requests.get
打了猴子补丁,这样我们就可以测试get_page
方法能否正确处理异常了,这样做既不需要为requests.get
构造能抛出异常的输入参数,也不需要改变requests.get
的源代码。
当我们需要对处理某个函数、类的函数进行测试时,被处理的函数和类可能需要精细的构造、长时间的运行,才能出现一些罕见的行为,为此我们通过猴子补丁直接得到这些情况导致的结果,从而测试外部的函数,当然猴子补丁并不只这一种用法
常见问题
- 当猴子补丁的假设由于代码更新不再为真时,这可能导致一些问题,因此猴子补丁常常在条件下使用
- 如果两个模块同时打了猴子补丁,只有后打的补丁才有效
- 源代码和其行为在不知道有猴子补丁的人的视角下会出现矛盾
参考文档
[1]https://stackoverflow.com/questions/5626193/what-is-monkey-patching
[2]https://en.wikipedia.org/wiki/Monkey_patch