python property的使用技巧分享
一種用起來像是使用實(shí)例屬性一樣的特殊屬性,可以對應(yīng)于某個方法
既要保護(hù)類的封裝特性,又要讓開發(fā)者可以使用 對象.屬性 的方式操作方法,@property 裝飾器,可以直接通過方法名來訪問方法,不需要在方法名后添加一對 () 小括號。
來看下求圓的面積的例子
class Circle(object): PI = 3.14 def __init__(self, r):# r圓的半徑self.r = rself.__area = self.PI * self.r * self.r@property def area(self):return self.__area def get_area(self):return self.__areaIn [2]: c = Circle(10)In [3]: c.areaOut[3]: 314.0In [4]: c.get_area()Out[4]: 314.0property屬性的定義和調(diào)用要注意一下幾點(diǎn): 定義時,在實(shí)例方法的基礎(chǔ)上添加 @property 裝飾器;并且僅有一個 self 參數(shù) 調(diào)用時,無需括號 ()
實(shí)例方法:c.get_area()
property裝飾的方法:c.area
具體實(shí)例對于某商城中顯示電腦主機(jī)的列表頁面,每次請求不可能把數(shù)據(jù)庫中的所有內(nèi)容都顯示到頁面上,而是通過分頁的功能局部顯示,所以在向數(shù)據(jù)庫中請求數(shù)據(jù)時就要顯示的指定獲取從第 m 條到第 n條的所有數(shù)據(jù) 這個分頁的功能包括:
根據(jù)用戶請求的當(dāng)前頁和總數(shù)據(jù)條數(shù)計算出 m 和 n 根據(jù) m 和 n 去數(shù)據(jù)庫中請求數(shù)據(jù)class Pager(object):def __init__(self, current_page):# 用戶當(dāng)前請求的頁碼(第一頁、第二頁...)self.current_page = current_page# 每頁默認(rèn)顯示10條數(shù)據(jù)self.per_items = 10 @property def start(self):val = (self.current_page - 1) * self.per_itemsreturn val @property def end(self):val = self.current_page * self.per_itemsreturn val# ipython測驗(yàn)In [2]: p = Pager(1)In [3]: p.start# 就是起始值,即:mOut[3]: 0In [4]: p.end# 就是結(jié)束值,即:nOut[4]: 10In [5]: p = Pager(2)In [6]: p.startOut[6]: 10In [7]: p.endOut[7]: 20property屬性的有兩種方式 裝飾器 即:在方法上應(yīng)用裝飾器 @property 類屬性 即:在類中定義值為 property 對象的類屬性 property() 裝飾器方式
在類的實(shí)例方法上應(yīng)用 @property 裝飾器
Python中的類有舊式類 和 新式類,新式類 的屬性比 舊式類的屬性豐富。
舊式類舊式類,具有一種 @property 裝飾器
class Goods:def __init__(self, name):self.name = name @property def price(self):return 100 # ipython測驗(yàn)In [10]: g = Goods(’手表’)In [11]: g.priceOut[11]: 100新式類
新式類,具有三種 @property 裝飾器
class Goods: ''' python3中默認(rèn)繼承object類 以python2、3執(zhí)行此程序的結(jié)果不同,因?yàn)橹挥性趐ython3中才有@xxx.setter @xxx.deleter ''' @property def price(self):print(’@property’) @price.setter def price(self, value):print(’@price.setter’) @price.deleter def price(self):print(’@price.deleter’)# ipython測驗(yàn)In [13]: g = Goods()In [14]: g.price@propertyIn [15]: g.price = 100@price.setterIn [16]: del g.price@price.deleter g.price 單獨(dú)調(diào)用自動執(zhí)行 @property 修飾的 price 方法,并獲取方法的返回值 g.price = 100 賦值自動執(zhí)行 @price.setter 修飾的 price 方法,并將 100 賦值給方法的參數(shù) del g.price 刪除自動執(zhí)行 @price.deleter 修飾的 price 方法 注意 舊式類中的屬性只有一種訪問方式,其對應(yīng)被 @property 修飾的方法 新式類中的屬性有三種訪問方式,并分別對應(yīng)了三個被@property、@方法名.setter、@方法名.deleter 修飾的方法
由于新式類中具有三種訪問方式,我們可以根據(jù)它們幾個屬性的訪問特點(diǎn),分別將三個方法定義為對同一個屬性:獲取、修改、刪除。
# Goods類@property應(yīng)用class Goods(object): def __init__(self, name, price):# 原價self.original_price = price# 折扣self.discount = 0.8 @property def price(self):# 實(shí)際價格 = 原價 * 折扣new_price = self.original_price * self.discountreturn new_price @price.setter def price(self, value):self.original_price = value @price.deleter def price(self):print(’刪除商品原價’)del self.original_price# ipython測驗(yàn)In [22]: g = Goods(’小米手機(jī)’, 2000)In [23]: g.priceOut[23]: 1600.0In [24]: g.price = 3000In [25]: g.priceOut[25]: 2400.0In [26]: del g.price刪除商品原價In [27]: g.price---------------------------------------------------------------------------AttributeError Traceback (most recent call last)<ipython-input-27-38ee45b469f2> in <module>----> 1 g.price<ipython-input-18-d5ea66eb7ece> in price(self) 12 def price(self): 13 # 實(shí)際價格 = 原價 * 折扣---> 14 new_price = self.original_price * self.discount 15 return new_price 16AttributeError: ’Goods’ object has no attribute ’original_price’類屬性方式
創(chuàng)建值為 property 對象的類屬性,當(dāng)使用類屬性的方式創(chuàng)建 property 屬性時,舊式類 和 新式類無區(qū)別
class Foo:def get_bar(self):return ’get_bar’ BAR = property(get_bar)# ipython 測驗(yàn)In [32]: f = Foo()In [33]: f.BAROut[33]: ’get_bar’
f.BAR 自動調(diào)用 get_bar() 方法,并獲取方法的返回值
property() 中有個四個參數(shù)
第一個參數(shù)是方法名,調(diào)用 對象.屬性 時自動觸發(fā)執(zhí)行方法 第二個參數(shù)是方法名,調(diào)用 對象.屬性 = XXX 時自動觸發(fā)執(zhí)行方法 第三個參數(shù)是方法名,調(diào)用 del 對象.屬性 時自動觸發(fā)執(zhí)行方法 第四個參數(shù)是字符串,調(diào)用 對象.屬性.__doc__ ,此參數(shù)是該屬性的描述信息class Foo(object): def __init__(self, bar):self.bar = bardef get_bar(self):print(’get_bar’)return self.bar def set_bar(self, value): '''必須要有兩個參數(shù)'''print(’set bar ’ + value)self.bar = value def del_bar(self):print(’del bar’)del self.bar BAR = property(get_bar, set_bar, del_bar, 'bar description...') # ipython測驗(yàn)In [50]: f = Foo(’python’)In [51]: f.BARget_barOut[51]: ’python’In [52]: f.BAR = ’Java’set bar JavaIn [53]: f.BARget_barOut[53]: ’Java’In [54]: del f.BARdel barproperty對象與@property裝飾器對比
由于 類屬性方式 創(chuàng)建 property 對象屬性具有3種訪問方式,我們可以根據(jù)它們幾個屬性的訪問特點(diǎn),分別將三個方法定義為對 同一個屬性:獲取、修改、刪除 ,跟 @property 裝飾器對比。
property對象類屬性# Goods類 property對象類屬性 應(yīng)用class Goods(object): def __init__(self, name, price):# 原價self.original_price = price# 折扣self.discount = 0.8 def get_price(self):# 實(shí)際價格 = 原價 * 折扣new_price = self.original_price * self.discountreturn new_price def set_price(self, value):self.original_price = value def del_price(self):print(’刪除商品原價’)del self.original_price PRICE = property(get_price, set_price, del_price, 'price description') # ipython測驗(yàn)In [59]: g = Goods(’Mac電腦’, 9000)In [60]: g.PRICEOut[60]: 7200.0In [61]: g.PRICE = 10000In [62]: g.PRICEOut[62]: 8000.0In [63]: del g.PRICE刪除商品原價@property裝飾器
# Goods類 @property裝飾器 應(yīng)用class Goods(object): def __init__(self, name, price):# 原價self.original_price = price# 折扣self.discount = 0.8 @property def price(self):# 實(shí)際價格 = 原價 * 折扣new_price = self.original_price * self.discountreturn new_price @price.setter def price(self, value):self.original_price = value @price.deleter def price(self):print(’刪除商品原價’)del self.original_price# ipython測驗(yàn)In [59]: g = Goods(’Mac電腦’, 9000)In [60]: g.PRICEOut[60]: 7200.0In [61]: g.PRICE = 10000In [62]: g.PRICEOut[62]: 8000.0In [63]: del g.PRICE刪除商品原價
可以發(fā)現(xiàn)兩種都可以實(shí)現(xiàn)但 @property 裝飾器的在 舊式類中只有 @property , 沒有@method.setter 和
@method.deleter,新式類則兩種都可以使用。因此看大家的習(xí)慣,選一種。
以上就是python property的使用技巧分享的詳細(xì)內(nèi)容,更多關(guān)于python property的資料請關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. 通過CSS數(shù)學(xué)函數(shù)實(shí)現(xiàn)動畫特效2. CSS3實(shí)例分享之多重背景的實(shí)現(xiàn)(Multiple backgrounds)3. CSS Hack大全-教你如何區(qū)分出IE6-IE10、FireFox、Chrome、Opera4. UDDI FAQs5. uni-app低成本封裝一個取色器組件的簡單方法6. 詳解CSS偽元素的妙用單標(biāo)簽之美7. 告別AJAX實(shí)現(xiàn)無刷新提交表單8. CSS3中Transition屬性詳解以及示例分享9. 解析原生JS getComputedStyle10. 使用css實(shí)現(xiàn)全兼容tooltip提示框
