函数是Python里的“一等公民”,可以像变量一样传递和引用,例如:
def hhh():
print("hhh")
def ddd():
print("ddd")
hhh=ddd
hhh()
输出:
ddd
hhh函数被ddd函数替换了。
而猴子补丁是一种替换类中函数的方法,可以动态地修改类中函数:
class Test:
def func(self):
print("hi")
def monkey(self):
print("hi monkey")
Test.func=monkey
a=Test()
a.func()
输出:
hi monkey
我们定义了猴子补丁monkey(也可以取别的名字)函数,接收一个参数,注意因为要替换的是类的实例方法,所以第一个参数是实例对象自身,一般取self,取别的名字也可以。monkey函数的参数个数也不一定需要和待替换的func函数一致,可以更多,但是要注意替换后调用时就要按monkey函数的参数数量传参数了。
class Test:
def func(self):
print("hi")
def monkey(self,a):#参数个数不一定要和func一致
print(self)#打印出<__main__.Test object at 0x7fb5cef7d5b0>,可以看到是类实例对象自身
print("hi monkey")
a=Test()
Test.func=monkey#替换可以发生在实例化之后
a.func(3)#因为monkey函数的参数个数增加了一个,所以要多传一个
输出:
<__main__.Test object at 0x7fb5cef7d5b0>
hi monkey
还可以替换实例中的func函数:
class Test:
def func(self):
print("hi")
def monkey(self,a):
print(self)
print("hi monkey")
a=Test()
a.func=monkey#替换类实例中的func函数
a.func(a,3)#但是这时就需要显式地把实例自身作为第一个参数传入
输出:
<__main__.Test object at 0x7f87ec2935b0>
hi monkey