1. 程式人生 > >python Django裝飾器@property的妙用

python Django裝飾器@property的妙用

@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
顯然,直接給屬性賦值無法檢查分數的有效性。

如果利用兩個方法:

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
    @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
>>> s.score = 1000
Traceback (most recent call last):
  ...
ValueError: invalid score
說明對 score 賦值實際呼叫的是 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

    ???

s = Student('Bob', 59)
print s.grade

s.score = 60
print s.grade

s.score = 99
print s.grade