根据廖雪峰的python教程
其中底下有一条评论总结的很好,拿来引用,原文
1、why @property?
绑定属性有两种方式,:
一是通过“实例.属性”。优点:简单;缺点:直接暴露、无法进行参数检查等。如:
s = Student()
s.score = 9999
优点:简单;缺点:直接暴露、无法进行参数检查等。
二是通过设置类方法进行绑定。优点:没有直接暴露、可以进行参数检查;缺点:调用不简单,如:
class Student(object):
def get_score(self):
return self._score
def set_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综合了两种优点:既能参数检查,还能类似于访问属性那样进行绑定。故使用@property对方式二的函数进行装饰,从而实现以上的优点。
2、@property使用?
getter方法---->属性:@property
setter方法---->属性赋值:@property修饰的@函数名.setter (这里的函数名是指property装饰的函数名)
从而实现可控的属性操作
只读:只定义getter方法,即只有@property
读写:定义getter、setter两个方法,即@property、@函数名.setter
3、读+写的模板如下(读的模板只用上面即可):
class 类名(object):
@property
def 函数名(self):
return self._函数名 #例如函数名是score,属性名则为_score,这样目的是属性设置为私有属性,且函数名到时类似于属性调用 时便于理解,例如s.score = 99 一看就知道是在设置分数,但我们要理解的时此时并非是访问属性,本质是在调用函数
@函数名.setter
def 函数名(self, value):
if 参数检查:
raise 错误
self._函数名 = value
4、特别注意:
属性的方法名不要和实例变量重名。例如,以下的代码是错误的:
class Student(object):
# 方法名称和实例变量均为birth:
@property
def birth(self):
return self.birth
这是因为调用s.birth
时,首先转换为方法调用,在执行return self.birth
时,又视为访问self
的属性,于是又转换为方法调用,造成无限递归,最终导致栈溢出报错RecursionError
。