Python 中@property的用法


在綁定屬性時,如果我們直接把屬性賦值給對象,比如:

p = Person()
p.name= 'Mary'

我們先看個詳細的例子(注意雙下划線name和age定義為私有變量):

class Person(object):
def __init__(self, name, age):
self.__name = name
self.__age = age

def get_age_fun(self):
return self.__age

def set_age_fun(self, value):
if not isinstance(value, int):
raise ValueError('年齡必須是數字!')
if value < 0 or value > 100:
raise ValueError('年齡必須是0-100')
self.__age = value

def print_info(self):
print('%s: %s' % (self.__name, self.__age))


p = Person('balala',20)
p.__age = 17
print(p.__age) # 17
print(p.get_age_fun()) # 20 表面上看,上面代碼“成功”地設置了__age變量 17,但實際上這個__age變量和class內部的__age變量不是一個變量!
# 內部的__age變量已經被Python解釋器自動改成了_Person_age,而外部代碼給p新增了一個__age變量。 所以調用 get_age_fun輸出的是初始值

p.set_age_fun(35)
print(p.get_age_fun()) # 35

print(p.print_info()) # balala: 35

輸出: 

17
20
35
balala: 35

表面上看,外部代碼“成功”地設置了__age變量 17,但實際上這個_age變量和class內部的_age變量不是一個變量!
內部的_age變量已經被Python解釋器自動改成了_Person_age,而外部代碼給p新增了一個_age變量。 所以調用 get_age_fun輸出的是初始值 20
而set_age_fun 通過class內部改變了age變量值,所以最終輸出 balala: 35

我們再稍微調整下:

(注意只改變了一個變量名: 原來的私有屬性 __age 單下划線為: _age,也可以定義為:age. 
解釋:以一個下划線開頭的實例變量名,比如_age,這樣的實例變量外部是可以訪問的,但是,按照約定俗成的規定,當看到這樣的變量時,意思是,"雖然可以被訪問,但是,請視為私有變量,不要隨意訪問。")
class Person(object):
    def __init__(self, name, age):
        self.__name = name
        self._age = age

    def get_age_fun(self):
         return self._age

    def set_age_fun(self, value):
        if not isinstance(value, int):
            raise ValueError('年齡必須是數字!')
        if value < 0 or value > 100:
            raise ValueError('年齡必須是0-100')
        self._age = value

    def print_info(self):
        print('%s: %s' % (self.__name, self._age))


p = Person('balala',20)
p._age = 17
print(p._age) # 17
print(p.get_age_fun()) # 這里是17 不再是 20,因為此時_age是全局變量,外部直接影響到類內部的更新值

p.set_age_fun(35)
print(p.get_age_fun()) # 35

print(p.print_info()) # balala: 35

輸出:

1 17
2 17
3 35
4 balala: 35

看的出私有和全局的設置

但是,上面的調用方法是不是略顯復雜,沒有直接用屬性這么直接簡單。

有沒有可以用類似屬性這樣簡單的方式來訪問類的變量呢?必須的,對於類的方法
我們先來看一個稍微改造的例子:(稍后我們再使用Python內置的@property裝飾器就是負責把一個方法變成屬性調用.)

我們進入正題:看看@property的妙用之處:

 1 class Person(object):
 2     def __init__(self, name, age):
 3         self.__name = name
 4         self.__age = age
 5 
 6     @property
 7     def get_age_fun(self):
 8          return self.__age
 9 
10     @get_age_fun.setter # get_age_fun是上面聲明的方法
11     def set_age_fun(self, value):
12         if not isinstance(value, int):
13             raise ValueError('年齡必須是數字!')
14         if value < 0 or value > 100:
15             raise ValueError('年齡必須是0-100')
16         self.__age = value
17 
18     def print_info(self):
19         print('%s: %s' % (self.__name, self.__age))
20 
21 
22 p = Person('balala',20)
23 p.__age = 17
24 print(p.__age) # 17
25 print(p.get_age_fun) # 20 注意這里不帶()
26 
27 #p.set_age_fun(35) 注意不能這樣調用賦值了
28 p.set_age_fun = 35 #  這里set_age_fun 就是 聲明的函數不帶()
29 print(p.get_age_fun) # 35
30 print(p.print_info()) # balala: 35

輸出:

17
20
35
balala: 35

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM