python 面向對象類成員(字段 方法 屬性)


一、字段

字段包括:普通字段和靜態字段,他們在定義和使用中有所區別,而最本質的區別是內存中保存的位置不同,

  • 普通字段屬於對象
  • 靜態字段屬於
class Province:
    # 靜態字段
    country = '中國'
    def __init__(self, name):
        # 普通字段
        self.name = name

# 直接訪問普通字段
obj = Province('河北省')
print obj.name

# 直接訪問靜態字段
Province.country

 

由上述代碼可以看出【普通字段需要通過對象來訪問】【靜態字段通過類訪問】,在使用上可以看出普通字段和靜態字段的歸屬是不同的。其在內容的存儲方式類似如下圖:

由上圖可是:

  • 靜態字段在內存中只保存一份
  • 普通字段在每個對象中都要保存一份

應用場景: 通過類創建對象時,如果每個對象都具有相同的字段,那么就使用靜態字段

二、方法

方法包括:普通方法、靜態方法和類方法,三種方法在內存中都歸屬於類,區別在於調用方式不同。

  • 普通方法:由對象調用;至少一個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()

 

相同點:對於所有的方法而言,均屬於類(非對象)中,所以,在內存中也只保存一份。

不同點:方法調用者不同、調用方法時自動傳入的參數不同。

三、屬性  

如果你已經了解Python類中的方法,那么屬性就非常簡單了,因為Python中的屬性其實是普通方法的變種。

對於屬性,有以下三個知識點:

  • 屬性的基本使用
  • 屬性的兩種定義方式

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的屬性的功能是:屬性內部進行一系列的邏輯計算,最終將計算結果返回。

2、屬性的兩種定義方式

屬性的定義有兩種方式:

  • 裝飾器 即:在方法上應用裝飾器
  • 靜態字段 即:在類中定義值為property對象的靜態字段

裝飾器方式:在類的普通方法上應用@property裝飾器

我們知道Python中的類有經典類和新式類,新式類的屬性比經典類的屬性豐富。( 如果類繼object,那么該類是新式類 )
經典類,具有一種@property裝飾器(如上一步實例)

# ############### 定義 ############### 
class Goods:

@property
def price(self):
  return "wupeiqi"
# ############### 調用 ###############
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     # 刪除商品原價

靜態字段方式,創建值為property對象的靜態字段

當使用靜態字段的方式創建屬性時,經典類和新式類無區別

class Foo:

def get_bar(self):
  return 'wupeiqi'

BAR = property(get_bar)

obj = Foo()
reuslt = obj.BAR # 自動調用get_bar方法,並獲取方法的返回值
print reuslt

 

property的構造方法中有個四個參數

  • 第一個參數是方法名,調用 對象.屬性 時自動觸發執行方法
  • 第二個參數是方法名,調用 對象.屬性 = XXX 時自動觸發執行方法
  • 第三個參數是方法名,調用 del 對象.屬性 時自動觸發執行方法
  • 第四個參數是字符串,調用 對象.屬性.__doc__ ,此參數是該屬性的描述信息
class Foo:

def get_bar(self):
  return 'wupeiqi'

# *必須兩個參數
def set_bar(self, value): 
  return return 'set value' + value

def del_bar(self):
  return 'wupeiqi'

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

 注意:Python WEB框架 Django 的視圖中 request.POST 就是使用的靜態字段的方式創建的屬性

class WSGIRequest(http.HttpRequest):
    def __init__(self, environ):
        script_name = get_script_name(environ)
        path_info = get_path_info(environ)
        if not path_info:
            # Sometimes PATH_INFO exists, but is empty (e.g. accessing
            # the SCRIPT_NAME URL without a trailing slash). We really need to
            # operate as if they'd requested '/'. Not amazingly nice to force
            # the path like this, but should be harmless.
            path_info = '/'
        self.environ = environ
        self.path_info = path_info
        self.path = '%s/%s' % (script_name.rstrip('/'), path_info.lstrip('/'))
        self.META = environ
        self.META['PATH_INFO'] = path_info
        self.META['SCRIPT_NAME'] = script_name
        self.method = environ['REQUEST_METHOD'].upper()
        _, content_params = cgi.parse_header(environ.get('CONTENT_TYPE', ''))
        if 'charset' in content_params:
            try:
                codecs.lookup(content_params['charset'])
            except LookupError:
                pass
            else:
                self.encoding = content_params['charset']
        self._post_parse_error = False
        try:
            content_length = int(environ.get('CONTENT_LENGTH'))
        except (ValueError, TypeError):
            content_length = 0
        self._stream = LimitedStream(self.environ['wsgi.input'], content_length)
        self._read_started = False
        self.resolver_match = None

    def _get_scheme(self):
        return self.environ.get('wsgi.url_scheme')

    def _get_request(self):
        warnings.warn('`request.REQUEST` is deprecated, use `request.GET` or '
                      '`request.POST` instead.', RemovedInDjango19Warning, 2)
        if not hasattr(self, '_request'):
            self._request = datastructures.MergeDict(self.POST, self.GET)
        return self._request

    @cached_property
    def GET(self):
        # The WSGI spec says 'QUERY_STRING' may be absent.
        raw_query_string = get_bytes_from_wsgi(self.environ, 'QUERY_STRING', '')
        return http.QueryDict(raw_query_string, encoding=self._encoding)
    
    # ############### 看這里看這里  ###############
    def _get_post(self):
        if not hasattr(self, '_post'):
            self._load_post_and_files()
        return self._post

    # ############### 看這里看這里  ###############
    def _set_post(self, post):
        self._post = post

    @cached_property
    def COOKIES(self):
        raw_cookie = get_str_from_wsgi(self.environ, 'HTTP_COOKIE', '')
        return http.parse_cookie(raw_cookie)

    def _get_files(self):
        if not hasattr(self, '_files'):
            self._load_post_and_files()
        return self._files

    # ############### 看這里看這里  ###############
    POST = property(_get_post, _set_post)
    
    FILES = property(_get_files)
    REQUEST = property(_get_request)

Django源碼
View Code

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


免責聲明!

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



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