1. 字段
字段:包括普通字段和靜態字段,他們在定義和使用中有所區別,而最本質的區別是內存中保存的位置不同。
普通字段屬於對象
靜態字段屬於類
字段的定義和使用:
class Province: # 靜態字段 country = '中國' def __init__(self, name): # 普通字段 self.name = name # 直接訪問普通字段 obj = Province('河北省') print obj.name # 直接訪問靜態字段 Province.country
由上述代碼可以看出普通字段需要通過對象來訪問,靜態字段通過類訪問,在使用上可以看出普通字段和靜態字段的歸屬是不同的。其在內容的存儲方式類似如下圖:
由上圖可知:
靜態字段在內存中只保存一份
普通字段在每個對象中都要保存一份
應用場景: 通過類創建對象時,如果每個對象都具有相同的字段,那么就使用靜態字段
2. 方法
方法包括:普通方法、靜態方法和類方法,三種方法在內存中都歸屬於類,區別在於調用方式不同。
普通方法:由對象調用;至少一個self參數;執行普通方法時,自動將調用該方法的對象賦值給self;
類方法:由類調用; 至少一個cls參數;執行類方法時,自動將調用該方法的類復制給cls;
靜態方法:由類調用;無默認參數。
方法的定義和使用:
class Foo: def __init__(self, name): self.name = name def ord_func(self): """ 定義普通方法,至少有一個self參數 """ # print self.name print '普通方法' @classmethod def class_func(cls): """ 定義類方法,至少有一個cls參數 """ print '類方法' @staticmethod def static_func(): """ 定義靜態方法 ,無默認參數""" print '靜態方法' # 調用普通方法 f = Foo() f.ord_func() # 調用類方法 Foo.class_func() # 調用靜態方法 Foo.static_func()
相同點:對於所有的方法而言,均屬於類(非對象)中,所以,在內存中也只保存一份。
不同點:方法調用者不同、調用方法時自動傳入的參數不同。
3. 屬性
如果你已經了解Python類中的方法,那么屬性就非常簡單了,因為Python中的屬性其實是普通方法的變種。
對於屬性,有以下兩個知識點:
屬性的基本使用
屬性的兩種定義方式
3.1 屬性的基本使用
################ 定義 ############### class Foo: def func(self): pass # 定義屬性 @property def prop(self): pass # ############### 調用 ############### foo_obj = Foo() foo_obj.func() foo_obj.prop #調用屬性
由屬性的定義和調用要注意一下幾點:
定義時,在普通方法的基礎上添加 @property 裝飾器;
定義時,屬性僅有一個self參數
調用時,無需括號
方法:foo_obj.func()
屬性:foo_obj.prop
注意:屬性存在意義是:訪問屬性時可以制造出和訪問字段完全相同的假象,屬性由方法變種而來,如果Python中沒有屬性,方法完全可以代替其功能。
實例:對於主機列表頁面,每次請求不可能把數據庫中的所有內容都顯示到頁面上,而是通過分頁的功能局部顯示,所以在向數據庫中請求數據時就要顯示的指定獲取從第m條到第n條的所有數據(即:limit m,n),這個分頁的功能包括:
根據用戶請求的當前頁和總數據條數計算出 m 和 n
根據m 和 n 去數據庫中請求數據
# ############### 定義 ############### class Pager: def __init__(self, current_page): # 用戶當前請求的頁碼(第一頁、第二頁...) self.current_page = current_page # 每頁默認顯示10條數據 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 # ############### 調用 ############### p = Pager(1) p.start 就是起始值,即:m p.end 就是結束值,即:n
從上述可見,Python的屬性的功能是:屬性內部進行一系列的邏輯計算,最終將計算結果返回。
3.2 屬性的兩種定義方式
屬性的定義有兩種方式:
裝飾器:在方法上應用裝飾器
靜態字段:在類中定義值為property對象的靜態字段
3.2.1 裝飾器方式
裝飾器方式:在類的普通方法上應用@property裝飾器
我們知道Python中的類有經典類和新式類,新式類的屬性比經典類的屬性豐富。( 如果類繼object,那么該類是新式類 )
經典類,具有一種@property裝飾器(如上一步實例)
# ############### 定義 ############### class Goods: @property def price(self): return "xhh" # ############### 調用 ############### obj = Goods() result = obj.price # 自動執行 @property 修飾的 price 方法,並獲取方法的返回值
新式類,具有三種@property裝飾器
# ############### 定義 ############### class Goods(object): @property def price(self): print '@property' @price.setter def price(self, value): print '@price.setter' @price.deleter def price(self): print '@price.deleter' # ############### 調用 ############### obj = Goods() obj.price # 自動執行 @property 修飾的 price 方法,並獲取方法的返回值 obj.price = 123 # 自動執行 @price.setter 修飾的 price 方法,並將 123 賦值給方法的參數 del obj.price # 自動執行 @price.deleter 修飾的 price 方法
注:經典類中的屬性只有一種訪問方式,其對應被 @property 修飾的方法
新式類中的屬性有三種訪問方式,並分別對應了三個被@property、@方法名.setter、@方法名.deleter修飾的方法
由於新式類中具有三種訪問方式,我們可以根據他們幾個屬性的訪問特點,分別將三個方法定義為對同一個屬性:獲取、修改、刪除
例子:
class Goods(object): def __init__(self): # 原價 self.original_price = 100 # 折扣 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.deltter def price(self, value): del self.original_price obj = Goods() obj.price # 獲取商品價格 obj.price = 200 # 修改商品原價 del obj.price # 刪除商品原價
3.2.2 靜態字段方式
靜態字段方式:創建值為property對象的靜態字段
當使用靜態字段的方式創建屬性時,經典類和新式類無區別
class Foo: def get_bar(self): return 'xhh' BAR = property(get_bar) obj = Foo() reuslt = obj.BAR # 自動調用get_bar方法,並獲取方法的返回值 print reuslt
property的構造方法中有個四個參數:
第一個參數是方法名,調用
對象.屬性
時自動觸發執行方法第二個參數是方法名,調用
對象.屬性 = XXX
時自動觸發執行方法第三個參數是方法名,調用
del 對象.屬性
時自動觸發執行方法第四個參數是字符串,調用
對象.屬性.__doc__
,此參數是該屬性的描述信息
class Foo: def get_bar(self): return 'xhh' # *必須兩個參數 def set_bar(self, value): return return 'set value' + value def del_bar(self): return 'xhh' BAR = property(get_bar, set_bar, del_bar, 'description...') obj = Foo() obj.BAR # 自動調用第一個參數中定義的方法:get_bar obj.BAR = "alex" # 自動調用第二個參數中定義的方法:set_bar方法,並將“alex”當作參數傳入 del Foo.BAR # 自動調用第三個參數中定義的方法:del_bar方法 obj.BAE.__doc__ # 自動獲取第四個參數中設置的值:description...
由於靜態字段方式創建屬性具有三種訪問方式,我們可以根據他們幾個屬性的訪問特點,分別將三個方法定義為對同一個屬性:獲取、修改、刪除
實例:
class Goods(object): def __init__(self): # 原價 self.original_price = 100 # 折扣 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, value): del self.original_price PRICE = property(get_price, set_price, del_price, '價格屬性描述...') obj = Goods() obj.PRICE # 獲取商品價格 obj.PRICE = 200 # 修改商品原價 del obj.PRICE # 刪除商品原價
所以,定義屬性共有兩種方式,分別是【裝飾器】和【靜態字段】,而【裝飾器】方式針對經典類和新式類又有所不同。