当我们定义了一个 class,创建了一个 class 的实例后,我们可以给该实例绑定任何属性和方法,也可以给类绑定任何属性和方法,这就是动态语言的灵活性。
给实例绑定属性和方法,每个实例之间新增的部分是互不干扰的。
1. 先来看一个方法:MethodType
如果我们在模块中定义一个方法,而不是在类中定义一个方法,那么此时这个方法并不是对象的实例方法,但是我们可以将这个方法作为属性赋值给对象的
某一个属性。比如下面这样:
class Student(object): def __init___(self, name, age): self.name = name self.age = age # 需要被绑定的方法 def run(self): print("this is run method ", self) stu = Student() stu.run = run stu.run("xxxxxx") # this is run method xxxxxx
当我们定义一个 class 的时候可以给这个类定义实例属性和实例方法,实例方法的特点:当实例方法被调用的时候Python对象会自动作为self参数传入
到方法的内部。但是在调用 run 方法的时候 stu 对象不会作为 self 参数传递到 run 方法内部,就好像是 staticmethod 方法一样。MethodType 可以解决
这个问题,这个方法可以让一个模块方法在被调用的时候自动传入调用对象作为 self 参数。语法如下:
anotherRef = types.MethodType(methodName, instance/className)
于是上面的代码可以这样改:
from types import MethodType class Student(object): def __init___(self, name, age): self.name = name self.age = age # 需要被绑定的方法 def run(self): print("this is run method ", self) stu = Student() stu.run = MethodType(run, stu) stu.run() # this is run method <__main__.Student object at 0x000001B453CEF370>
2. 下面我们来介绍下如何动态绑定
1) 绑定方法到实例中
from types import MethodType class Student(object): pass def set_age(self, age): self.age = age stu1 = Student() stu2 = Student() stu1.set_age = MethodType(set_age, stu1) stu2.set_age = MethodType(set_age, stu2) stu1.set_age(22) stu2.set_age(44) print(stu1.age) # 22 print(stu2.age) # 44
2)绑定方法到类上
用 MethodType 将方法绑定到类,并不是将这个方法直接写到类内部,而是在内存中创建一个link指向外部的方法,在创建实例的时候这个link也会被复制。
from types import MethodType class Student (object): pass def set_age(self, age): self.age = age Student.set_age = MethodType(set_age, Student) stu1 = Student() stu2 = Student() stu1.set_age(23) stu2.set_age(99) print(stu1.age) # 99 print(stu2.age) # 99 print(id(stu1.set_age), id(stu2.set_age)) # 3022011453632 3022011453632 print(id(stu1.age), id(stu2.age)) # 140718085415648 140718085415648
为什么 stu1,stu2 都显示的 99 呢?,因为 stu1 和 stu2 都指向了相同的 set_age 和 age 地址。