@unique 装饰器的作用
@unique 装饰器是为了枚举类所做的类装饰器,它会搜索枚举类中枚举值,一旦发现有重复的值,就会抛出异常ValueError
class Mistake(Enum):ONE = 1TWO = 2THREE = 3FOUR = 3
ValueError: duplicate values found in <enum 'Mistake'>: FOUR -> THREE
@property 装饰器的作用
@property 装饰器是负责把一个方法变成属性调用的
class Student(object):@propertydef score(self):return self._score@score.setterdef score(self, value):if not isinstance(value, int):raise ValueError('score must be an integer!')if value < 0 or value > 100:raise ValueError('score must between 0 ~ 100!')self._score = value
@property 的实现比较复杂。只要对于一个getter方法加上@property,其就变成了属性,在上面的例子中,score()这个getter方法被加上@property后,就变成了变量一般的存在,只不过只是拥有get_score这个属性。同时,@property本身又创建了另一个装饰器,用以把setter方法也变成属性,在上面的例子中就是自动创建了装饰器@score.setter,于是方法score(value)就变成了变量一般的存在,只不过只是拥有set_score这个属性。
>>> s = Student()>>> s.score = 60 # OK,实际转化为s.set_score(60)>>> s.score # OK,实际转化为s.get_score()60>>> s.score = 9999Traceback (most recent call last):...ValueError: score must between 0 ~ 100!
还可以定义只读属性,只定义getter方法,不定义setter方法就是一个只读属性:
class Student(object):@propertydef birth(self):return self._birth@birth.setterdef birth(self, value):self._birth = value@propertydef age(self):return 2015 - self._birth
上面的birth是可读写属性,而age就是一个只读属性,因为age可以根据birth和当前时间计算出来。
@property 广泛应用在类的定义中,可以让调用者写出简短的代码,同时保证对参数进行必要的检查,这样,程序运行时就减少了出错的可能性。
@classmethod 装饰器的作用
classmethod 修饰符对应的函数不需要实例化,不需要 self 参数,但第一个参数需要是表示自身类的 cls 参数,可以来调用类的属性,类的方法,实例化对象等。
#!/usr/bin/python# -*- coding: UTF-8 -*-class A(object):# 属性默认为类属性(可以给直接被类本身调用)num = "类属性"# 实例化方法(必须实例化类之后才能被调用)def func1(self): # self : 表示实例化类后的地址idprint("func1")print(self)# 类方法(不需要实例化类就可以被类本身调用)@classmethoddef func2(cls): # cls : 表示没用被实例化的类本身print("func2")print(cls)print(cls.num)cls().func1()# 不传递传递默认self参数的方法(该方法也是可以直接被类调用的,但是这样做不标准)def func3():print("func3")print(A.num) # 属性是可以直接用类本身调用的# A.func1() 这样调用是会报错:因为func1()调用时需要默认传递实例化类后的地址id参数,如果不实例化类是无法调用的A.func2()A.func3()
相当于C++类中的static成员函数,可以在不实例化类的情况下直接调用类的static成员函数
参考资料
https://stackoverflow.com/questions/56139511/what-does-the-unique-decorator-do-in-python
