考察 Student 類:編程
class Student(object): def __init__(self, name, score): self.name = name self.score = score
當咱們想要修改一個 Student 的 scroe 屬性時,能夠這麼寫:函數
s = Student('Bob', 59) s.score = 60
可是也能夠這麼寫:spa
s.score = 1000
顯然,直接給屬性賦值沒法檢查分數的有效性。code
若是利用兩個方法:對象
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) 就會報錯。blog
這種使用 get/set 方法來封裝對一個屬性的訪問在許多面向對象編程的語言中都很常見。get
可是寫 s.get_score() 和 s.set_score() 沒有直接寫 s.score 來得直接。產品
有沒有一箭雙鵰的方法?----有。it
由於Python支持高階函數,能夠用裝飾器函數把 get/set 方法「裝飾」成屬性調用:面向對象編程
class Student(object): def __init__(self, name, score): self.name = name self.__score = score @property def score(self): return self.__score @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('Bob', 59) >>> s.score = 60 >>> print s.score 60 >>> s.score = 1000 Traceback (most recent call last): ... ValueError: invalid score