动态类装饰器函数在调用时读取,而静态类装饰器函数在导入时读取。这种差别引申出了两种类装饰器的不用适用场景。下面通过示例代码来直观的演示两种装饰器的区别。
静态类
class Test:
list = ['0']
@staticmethod
def out():
print(Test.list)
class child1(Test):
list = ['1', '2']
class child2(Test):
list = ['1', '5']
a = child1()
a.out()
b = child2()
b.out()
上述代码执行后,a.out()和b.out()输出结果都是[‘0’]。原因就是静态类函数out()调用的是父类Test的list,所以通过子类给list赋值的做法是行不通的,若想在子类改变out()的输出就必须在子类里重定义out()函数(见第三个示例)。通过在子类里给父类的list重新赋值的做法一样行不通,参见第四个示例。
动态类
class Test:
list = []
@classmethod
def out(cls):
print(cls.list)
class child1(Test):
list = ['1', '2']
class child2(Test):
list = ['1', '5']
a = child1()
a.out()
b = child2()
b.out()
上述代码执行后,a.out()输出结果为[‘1’, ‘2’],b.out()输出结果为[‘1’, ‘5’]。因为动态类函数指向的是子类本身的list,所以在子类里给list赋值后,out()函数输出的就是子类自己的list。
第三个示例
class Test:
list = ['0']
@staticmethod
def out():
print(Test.list)
class child1(Test):
list = ['1', '2']
@staticmethod
def out():
print(child1.list)
class child2(Test):
list = ['1', '5']
@staticmethod
def out():
print(child2.list)
child1.out()
child2.out()
该示例输出结果是[‘1’, ‘2’]和[‘1’, ‘5’]。
第四个示例
class Test:
list = []
@staticmethod
def out():
print(Test.list)
class child1(Test):
Test.list = ['1', '5']
class child2(Test):
Test.list = ['1', '2']
child1.out()
该示例输出结果为[‘1’, ‘2’],什么原因?为什么不是[‘1’, ‘5’]?因为子类本身没有out()函数,向上查找到父类Test有该函数,但该函数为静态函数,在文件载入时就已经读取至内存,读取过程中搜索Test.list变量的值,于是处于最后面的child2里的变量值便被采用。