Python中關(guān)于property使用的小技巧
property屬性
一種用起來像是使用實例屬性一樣的特殊屬性,可以對應(yīng)于某個方法
既要保護類的封裝特性,又要讓開發(fā)者可以使用 對象.屬性 的方式操作方法,@property 裝飾器,可以直接通過方法名來訪問方法,不需要在方法名后添加一對 () 小括號。
來看下求圓的面積的例子
class Circle(object): PI = 3.14 def __init__(self, r): # r圓的半徑 self.r = r self.__area = self.PI * self.r * self.r @property def area(self): return self.__area def get_area(self): return self.__area In [2]: c = Circle(10) In [3]: c.area Out[3]: 314.0 In [4]: c.get_area() Out[4]: 314.0
property屬性的定義和調(diào)用要注意一下幾點:
- 定義時,在實例方法的基礎(chǔ)上添加
@property裝飾器;并且僅有一個self參數(shù) - 調(diào)用時,無需括號
()
實例方法:c.get_area() property裝飾的方法:c.area
具體實例
對于某商城中顯示電腦主機的列表頁面,每次請求不可能把數(shù)據(jù)庫中的所有內(nèi)容都顯示到頁面上,而是通過分頁的功能局部顯示,所以在向數(shù)據(jù)庫中請求數(shù)據(jù)時就要顯示的指定獲取從第 m 條到第 n條的所有數(shù)據(jù) 這個分頁的功能包括:
- 根據(jù)用戶請求的當前頁和總數(shù)據(jù)條數(shù)計算出 m 和 n
- 根據(jù) m 和 n 去數(shù)據(jù)庫中請求數(shù)據(jù)
class Pager(object): def __init__(self, current_page): # 用戶當前請求的頁碼(第一頁、第二頁...) self.current_page = current_page # 每頁默認顯示10條數(shù)據(jù) self.per_items = 10 @property def start(self): val = (self.current_page - 1) * self.per_items return val @property def end(self): val = self.current_page * self.per_items return val # ipython測驗 In [2]: p = Pager(1) In [3]: p.start # 就是起始值,即:m Out[3]: 0 In [4]: p.end # 就是結(jié)束值,即:n Out[4]: 10 In [5]: p = Pager(2) In [6]: p.start Out[6]: 10 In [7]: p.end Out[7]: 20
property屬性的有兩種方式
- 裝飾器 即:在方法上應(yīng)用裝飾器
@property - 類屬性 即:在類中定義值為
property對象的類屬性property()
裝飾器方式
在類的實例方法上應(yīng)用 @property 裝飾器
Python中的類有舊式類 和 新式類,新式類 的屬性比 舊式類的屬性豐富。
舊式類
舊式類,具有一種 @property 裝飾器
class Goods:
def __init__(self, name):
self.name = name
@property
def price(self):
return 100
# ipython測驗
In [10]: g = Goods('手表')
In [11]: g.price
Out[11]: 100
新式類
新式類,具有三種 @property 裝飾器
class Goods:
"""
python3中默認繼承object類
以python2、3執(zhí)行此程序的結(jié)果不同,因為只有在python3中才有@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測驗
In [13]: g = Goods()
In [14]: g.price
@property
In [15]: g.price = 100
@price.setter
In [16]: del g.price
@price.deleter
g.price單獨調(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ù)它們幾個屬性的訪問特點,分別將三個方法定義為對同一個屬性:獲取、修改、刪除。
# Goods類@property應(yīng)用
class Goods(object):
def __init__(self, name, price):
# 原價
self.original_price = price
# 折扣
self.discount = 0.8
@property
def price(self):
# 實際價格 = 原價 * 折扣
new_price = self.original_price * self.discount
return new_price
@price.setter
def price(self, value):
self.original_price = value
@price.deleter
def price(self):
print('刪除商品原價')
del self.original_price
# ipython測驗
In [22]: g = Goods('小米手機', 2000)
In [23]: g.price
Out[23]: 1600.0
In [24]: g.price = 3000
In [25]: g.price
Out[25]: 2400.0
In [26]: del g.price
刪除商品原價
In [27]: g.price
---------------------------------------------------------------------------
AttributeErrorTraceback (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# 實際價格 = 原價 * 折扣
---> 14new_price = self.original_price * self.discount
15return new_price
16
AttributeError: 'Goods' object has no attribute 'original_price'
類屬性方式
創(chuàng)建值為 property 對象的類屬性,當使用類屬性的方式創(chuàng)建 property 屬性時,舊式類 和 新式類無區(qū)別
class Foo: def get_bar(self): return 'get_bar' BAR = property(get_bar) # ipython 測驗 In [32]: f = Foo() In [33]: f.BAR Out[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 = bar
def 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測驗
In [50]: f = Foo('python')
In [51]: f.BAR
get_bar
Out[51]: 'python'
In [52]: f.BAR = 'Java'
set bar Java
In [53]: f.BAR
get_bar
Out[53]: 'Java'
In [54]: del f.BAR
del bar
property對象與@property裝飾器對比
由于 類屬性方式 創(chuàng)建 property 對象屬性具有3種訪問方式,我們可以根據(jù)它們幾個屬性的訪問特點,分別將三個方法定義為對 同一個屬性:獲取、修改、刪除 ,跟 @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):
# 實際價格 = 原價 * 折扣
new_price = self.original_price * self.discount
return 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測驗
In [59]: g = Goods('Mac電腦', 9000)
In [60]: g.PRICE
Out[60]: 7200.0
In [61]: g.PRICE = 10000
In [62]: g.PRICE
Out[62]: 8000.0
In [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):
# 實際價格 = 原價 * 折扣
new_price = self.original_price * self.discount
return new_price
@price.setter
def price(self, value):
self.original_price = value
@price.deleter
def price(self):
print('刪除商品原價')
del self.original_price
# ipython測驗
In [59]: g = Goods('Mac電腦', 9000)
In [60]: g.PRICE
Out[60]: 7200.0
In [61]: g.PRICE = 10000
In [62]: g.PRICE
Out[62]: 8000.0
In [63]: del g.PRICE
刪除商品原價
可以發(fā)現(xiàn)兩種都可以實現(xiàn)但 @property 裝飾器的在 舊式類中只有 @property , 沒有@method.setter 和@method.deleter,新式類則兩種都可以使用。因此看大家的習(xí)慣,選一種。
大自然用數(shù)百億年創(chuàng)造出我們現(xiàn)實世界,而程序員用幾百年創(chuàng)造出一個完全不同的虛擬世界。我們用鍵盤敲出一磚一瓦,用大腦構(gòu)建一切。人們把1000視為權(quán)威,我們反其道行之,捍衛(wèi)1024的地位。我們不是鍵盤俠,我們只是平凡世界中不凡的締造者 。
到此這篇關(guān)于Python中關(guān)于property使用的小技巧的文章就介紹到這了,更多相關(guān)Python property 內(nèi)容請搜索本站以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持本站!
版權(quán)聲明:本站文章來源標注為YINGSOO的內(nèi)容版權(quán)均為本站所有,歡迎引用、轉(zhuǎn)載,請保持原文完整并注明來源及原文鏈接。禁止復(fù)制或仿造本網(wǎng)站,禁止在非maisonbaluchon.cn所屬的服務(wù)器上建立鏡像,否則將依法追究法律責(zé)任。本站部分內(nèi)容來源于網(wǎng)友推薦、互聯(lián)網(wǎng)收集整理而來,僅供學(xué)習(xí)參考,不代表本站立場,如有內(nèi)容涉嫌侵權(quán),請聯(lián)系alex-e#qq.com處理。
關(guān)注官方微信