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('年齡必須是數(shù)字!') 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 表面上看,上面代碼“成功”地設(shè)置了__age變量 17,但實際上這個__age變量和class內(nèi)部的__age變量不是一個變量! # 內(nèi)部的__age變量已經(jīng)被Python解釋器自動改成了_Person_age,而外部代碼給p新增了一個__age變量。 所以調(diào)用 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
表面上看,外部代碼“成功”地設(shè)置了__age變量 17,但實際上這個_age變量和class內(nèi)部的_age變量不是一個變量!
內(nèi)部的_age變量已經(jīng)被Python解釋器自動改成了_Person_age,而外部代碼給p新增了一個_age變量。 所以調(diào)用 get_age_fun輸出的是初始值 20
而set_age_fun 通過class內(nèi)部改變了age變量值,所以最終輸出 balala: 35
我們再稍微調(diào)整下:
(注意只改變了一個變量名: 原來的私有屬性 __age 單下劃線為: _age,也可以定義為:age.
解釋:以一個下劃線開頭的實例變量名,比如_age,這樣的實例變量外部是可以訪問的,但是,按照約定俗成的規(guī)定,當看到這樣的變量時,意思是,"雖然可以被訪問,但是,請視為私有變量,不要隨意訪問。")
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('年齡必須是數(shù)字!') 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是全局變量,外部直接影響到類內(nèi)部的更新值 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
看的出私有和全局的設(shè)置
但是,上面的調(diào)用方法是不是略顯復(fù)雜,沒有直接用屬性這么直接簡單。
有沒有可以用類似屬性這樣簡單的方式來訪問類的變量呢?必須的,對于類的方法
我們先來看一個稍微改造的例子:(稍后我們再使用Python內(nèi)置的@property裝飾器就是負責把一個方法變成屬性調(diào)用.)
我們進入正題:看看@property的妙用之處:
class Person(object): def __init__(self, name, age): self.__name = name self.__age = age @property def get_age_fun(self): return self.__age @get_age_fun.setter # get_age_fun是上面聲明的方法 def set_age_fun(self, value): if not isinstance(value, int): raise ValueError('年齡必須是數(shù)字!') 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 注意這里不帶() #p.set_age_fun(35) 注意不能這樣調(diào)用賦值了 p.set_age_fun = 35 # 這里set_age_fun 就是 聲明的函數(shù)不帶() print(p.get_age_fun) # 35 print(p.print_info()) # balala: 35
輸出:
17
20
35
balala: 35
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python實現(xiàn)監(jiān)控屏幕界面內(nèi)容變化并發(fā)送通知
這篇文章主要為大家詳細介紹了如何利用Python實現(xiàn)實時監(jiān)控屏幕上的信息是否發(fā)生變化并發(fā)送通知,文中的示例代碼講解詳細,感興趣的可以了解一下2023-04-04Python while、for、生成器、列表推導(dǎo)等語句的執(zhí)行效率測試
這篇文章主要介紹了Python while、for、生成器、列表推導(dǎo)等語句的執(zhí)行效率測試,本文分別用兩段程序測算出了各語句的執(zhí)行效率,然后總結(jié)了什么情況下使用什么語句優(yōu)先使用的語句等,需要的朋友可以參考下2015-06-06Python Django中間件,中間件函數(shù),全局異常處理操作示例
這篇文章主要介紹了Python Django中間件,中間件函數(shù),全局異常處理操作,結(jié)合實例形式分析了Django中間件,中間件函數(shù),全局異常處理相關(guān)操作技巧,需要的朋友可以參考下2019-11-11