@unique 装饰器的作用
@unique 装饰器是为了枚举类所做的类装饰器,它会搜索枚举类中枚举值,一旦发现有重复的值,就会抛出异常ValueError
class Mistake(Enum):
ONE = 1
TWO = 2
THREE = 3
FOUR = 3
ValueError: duplicate values found in <enum 'Mistake'>: FOUR -> THREE
@property 装饰器的作用
@property 装饰器是负责把一个方法变成属性调用的
class Student(object):
@property
def score(self):
return self._score
@score.setter
def 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 = 9999
Traceback (most recent call last):
...
ValueError: score must between 0 ~ 100!
还可以定义只读属性,只定义getter方法,不定义setter方法就是一个只读属性:
class Student(object):
@property
def birth(self):
return self._birth
@birth.setter
def birth(self, value):
self._birth = value
@property
def 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 : 表示实例化类后的地址id
print("func1")
print(self)
# 类方法(不需要实例化类就可以被类本身调用)
@classmethod
def 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