python基礎(26):類的成員(字段、方法、屬性)


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     # 刪除商品原價

所以,定義屬性共有兩種方式,分別是【裝飾器】和【靜態字段】,而【裝飾器】方式針對經典類和新式類又有所不同。


免責聲明!

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



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