Discuss / Python / 总结总结总结

总结总结总结

Topic source

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

/oh Vesuvio

#2 Created at ... [Delete] [Delete and Lock User]

牛逼,清晰

小糊涂

#3 Created at ... [Delete] [Delete and Lock User]

一路看着总结跟过来

不长到140斤不改名字

咔咔門

#4 Created at ... [Delete] [Delete and Lock User]

第三点中注释里的说明,您说是设置私有属性,私以为只是加了一个下划线避免 名字重复造成的无线递归


  • 1

Reply