@property
考察 Student 類:
class Student(object):
def __init__(self, name, score):
self.name = name
self.score = score
當我們想要修改一個 Student 的 scroe 屬性時,可以這么寫:
s = Student('Bob', 59)
s.score = 60
但是也可以這么寫:
s.score = 1000
顯然,直接給屬性賦值無法檢查分數(shù)的有效性。
如果利用兩個方法:
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 方法來封裝對一個屬性的訪問在許多面向?qū)ο缶幊痰恼Z言中都很常見。
但是寫 s.get_score() 和 s.set_score() 沒有直接寫 s.score 來得直接。
有沒有兩全其美的方法?----有。
因為Python支持高階函數(shù),在函數(shù)式編程中我們介紹了裝飾器函數(shù),可以用裝飾器函數(shù)把 get/set 方法“裝飾”成屬性調(diào)用:
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裝飾后的副產(chǎn)品。
現(xiàn)在,就可以像使用屬性一樣設(shè)置score了:
>>> s = Student('Bob', 59)
>>> s.score = 60
>>> print s.score
>>> s.score = 1000
Traceback (most recent call last):
...
ValueError: invalid score
說明對 score 賦值實際調(diào)用的是 set方法。
任務
如果沒有定義set方法,就不能對“屬性”賦值,這時,就可以創(chuàng)建一個只讀“屬性”。
請給Student類加一個grade屬性,根據(jù) score 計算 A(>=80)、B、C(<60)。
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 ???s = Student('Bob', 59)print s.grades.score = 60print s.grades.score = 99print s.grade
裝飾器果然不是我能理解的,以后慢慢看吧。來自神奇的解釋性語言python。
聯(lián)系客服