@property的使用

考察 Student 类:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.score = score

当我们想要修改一个 Student 的 scroe 属性时,可以这么写:

s = Student('Ajax', 59)
s.score = 60

但是也可以这么写:

s.score = 100

显然,直接给属性赋值无法检查分数的有效性。

如果利用两个方法:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score
    def get_score(self):
        return self.__score
    def set_score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

这样一来,s.set_score(1000) 就会报错。

这种使用 get/set 方法来封装对一个属性的访问在许多面向对象编程的语言中都很常见。

但是写 s.get_score()s.set_score() 没有直接写 s.score 来得直接。

有没有两全其美的方法?----有。

因为Python支持高阶函数,可以用装饰器函数把 get/set 方法“装饰”成属性调用:

class Student(object):
    def __init__(self, name, score):
        self.name = name
        self.__score = score

    # score转换为属性,外部可是直接s.score访问
    @property 
    def score(self):
        return self.__score
    
    # @property本身又创建了另一个装饰器@score.setter,负责把一个setter方法变成属性赋值
    @score.setter 
    def score(self, score):
        if score < 0 or score > 100:
            raise ValueError('invalid score')
        self.__score = score

注意: 第一个 score(self) 是get方法,用 @property 装饰,第二个 score(self, score) 是set方法,用 @score.setter 装饰,@score.setter 是前一个 @property 装饰后的副产品。

现在,就可以像使用属性一样设置score了:

s = Student('Ajax', 59)
s.score = 60
print s.score
60
s.score = 1000
Traceback (most recent call last):
  ...
ValueError: invalid score

说明对 score 赋值实际调用的是 set方法。

posted @ 2019-09-16 11:03  久末丶  阅读(178)  评论(0编辑  收藏  举报