Python 的描述器Descriptors

Python 的描述器Descriptors

描述器的表現
—————————————————————————
用到3個魔術方法: __get__()\__set__()\__delete__()

方法簽名:
object.__get__(self, instance, owner)
object.__set__(self, instance, value)
object.__delete__(self, instance)

self 指當前實例,調用者
instance是owner的實例
owner是屬性的所屬的類

描述器定義
——————————————————————————
Python中,一個類實現了__get__()\__set__()\__delete__()中的任一種方法,就是描述器。

非數據描述符:僅實現__get__
數據描述符:實現__get__ \ __set__

如果一個類的類屬性設置為描述器,它被稱為owner屬主

屬性的訪問順序
——————————————————————————
class A:
def __init__(self):
self.a1 = ‘a1’
print(‘A init’)

def __get__(self, instance, owner):
print(“A.__get__{} {} {}”.format(self, instance, owner))
return self

def __set__(self, instance, value):
print(“A.__set__{} {} {}”.format(self , instance , value))
self.data = value

def __delete__(self, instance):
print(“A.__delete__{} {}”.format(self , instance))
del self

class B:
x = A()
def __init__(self):
print(‘B init’)
self.y = ‘y1’
self.x = ‘b.x’ #增加實例屬性

print(‘-‘*20)
print(B.x)
print(B.x.a1)

print(‘=’*20)
b = B()
print(b.x)
print(b.x.data)
print(b.__dict__)

b.x.a1 = 200

print(b.x.__dict__)
print(b.__dict__)

del b.x
print(b.x)
print(b.x.a1)
print(b.__dict__)

print(‘+’*20)

b.x = 300 #調用數據數據描述符的__set__方法,或非數據描述器的實例覆蓋
B.x = 600 #賦值即定義,覆蓋類屬性
print(b.x)
print(b.__dict__)
print(B.__dict__)

屬性查找順序:
實例的__dict__優先于非數據描述符
數據描述符優先于實例的__dict__ (屬性查找變成了描述器的__set__)

Python中的描述器應用
————————————————————————————
staticmethod() 和 classmethod()是非數據描述器。實例可以重新定義和覆蓋方法。

property()是數據描述符。實例不能覆蓋屬性的行為。

class staticmethod:
def __init__(self, fn):
self._fn = fn
print(‘1 init’)

def __get__(self, instance, owner):
print(“1.__get__{} {} {}”.format(self, instance, owner))
return self._fn

class classmethod:
def __init__(self , fn):
self._fn = fn
print(‘2 init’)

def __get__(self, instance , owner):
from functools import partial
print(“2.__get__{} {} {}”.format(self , instance , owner))
return partial(self._fn, owner)

class property:
def __init__(self, fn):
self._fn = fn
self.setfn = None
print(‘3 init’)

def __get__(self, instance, owner):
print(“3.__get__{} {} {}”.format(self , instance , owner))
return self._fn(instance)

def __set__(self, instance, value):
print(“3.__set__{} {} {}”.format(self , instance , value))
self.setfn(instance,value)

# def getter(self):
# return self.__get__

def setter(self,fn):
self.setfn = fn
return self

class B:
@staticmethod # add = staticmethod(add) -> add
def add(x, y):
return x + y

@classmethod # add = staticmethod(add) -> add
def add(cls, x, y):
return x + y

@property # x = property(x) -> x
def x(self):
return self._x

@x.setter # x = x.setter(x) -> x
def x(self, value):
self._x = value

def __init__(self):
self._x = 6

b = B()
# print(b.add(3, 5))
# print(b.__dict__)
# print(B.__dict__)

print(b.x)
b.x = 10
print(b.x)

本文來自投稿,不代表Linux運維部落立場,如若轉載,請注明出處:http://www.www58058.com/99690

(0)
JacoJaco
上一篇 2018-05-28
下一篇 2018-05-29

相關推薦

  • Python 部分知識點總結(十)

    此篇博客只是記錄第十二周未掌握或不熟悉的知識點,用來加深印象。

    Python筆記 2018-05-28
  • 封裝與解構 集合

    封裝和解構 封裝:將多個值進行分割,結合在一起,本質上返回元組,只是省掉了小括號 ‘==‘意思為內容一致,‘=’意思為內存空間一致 解構:把線性結構的元素解開,并順序的賦值給其他變量,左邊接納的變量數要和左邊解開的元素數量一致 集合不是非線性 解構中使用*變量名接收,但不能單獨使用,被*變量名收集后組成一個列表 第一個下劃線為9,結果被第二個下劃線重新賦值為…

    Python筆記 2018-04-01
  • 函數

    函數 數學定義:y=f(x),y是x的函數,x是自變量 Python函數 有若干個語句塊,函數名稱,參數列表構成,它是組織代碼的最小單元 完成一定作用 函數的作用 結構化編程對代碼的最基本的封裝,一般按照功能組織一段代碼 封裝的目的為了復用,減少了冗余代碼 代碼更加簡潔美觀,更加易讀 函數的分類 內建函數,如max(),reversed()等 庫函數,如ma…

    2018-04-16
  • 函數與生成器

    函數 由若干語句組成的語句塊,函數名,參數列表構成,是組織代碼的最小單元,完成一定的功能,結構化編程對代碼的最基本的封裝,封裝的目的是為了復用 函數中沒有return,隱式會返回一個none值 定義中的參數列表成為形式參數,只是一種形式表達,簡稱形參 調用時寫的參數是實際參數,是實實在在傳入的值,簡稱實參 函數的定義,只是聲明了參數,不會執行,可以進行調用,…

    2018-04-16
  • Python高階函數及裝飾器

    First Class Object 函數在Python中是一等公民 函數也是對象,可調用的對象 函數可作為普通變量、參數、返回值等 高階函數 數學定義:y=g(f(x)) 高階函數需滿足的條件,至少其一 接受一個或多個函數作為參數 輸出一個函數 內建函數的高階函數 排序:sorted(iterable[,key][,reverse]) 返回一個新列表,對一…

    2018-04-22
  • Python遞歸函數、生成器以及匿名函數

    Python遞歸函數、生成器以及匿名函數

    2018-04-15
欧美性久久久久