成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Python黑魔法之描述符

開發 后端
Descriptors(描述符)是Python語言中一個深奧但很重要的一個黑魔法,它被廣泛應用于Python語言的內核,熟練掌握描述符將會為Python程序員的工具箱添加一個額外的技巧。本文我將講述描述符的定義以及一些常見的場景,并且在文末會補充一下__getattr,__getattribute__, __getitem__這三個同樣涉及到屬性訪問的魔術方法。

引言

Descriptors(描述符)是Python語言中一個深奧但很重要的一個黑魔法,它被廣泛應用于Python語言的內核,熟練掌握描述符將會為Python程序員的工具箱添加一個額外的技巧。本文我將講述描述符的定義以及一些常見的場景,并且在文末會補充一下__getattr__getattribute____getitem__這三個同樣涉及到屬性訪問的魔術方法

描述符的定義

descr__get__(self, obj, objtype=None) --> value
descr.__set__(self, obj, value) --> None
descr.__delete__(self, obj) --> None

只要一個object attribute(對象屬性)定義了上面三個方法中的任意一個,那么這個類就可以被稱為描述符類。

描述符基礎

下面這個例子中我們創建了一個RevealAcess類,并且實現了__get__方法,現在這個類可以被稱為一個描述符類。

class RevealAccess(object):
    def __get__(self, obj, objtype):
        print('self in RevealAccess: {}'.format(self))
        print('self: {}\nobj: {}\nobjtype: {}'.format(self, obj, objtype))
class MyClass(object):
    x = RevealAccess()
    def test(self):
        print('self in MyClass: {}'.format(self))

EX1實例屬性

接下來我們來看一下__get__方法的各個參數的含義,在下面這個例子中,self即RevealAccess類的實例x,obj即MyClass類的實例m,objtype顧名思義就是MyClass類自身。從輸出語句可以看出,m.x訪問描述符x會調用__get__方法。

>>> m = MyClass()
>>> m.test()
self in MyClass: <__main__.MyClass object at 0x7f19d4e42160>
>>> m.x
self in RevealAccess: <__main__.RevealAccess object at 0x7f19d4e420f0>
self: <__main__.RevealAccess object at 0x7f19d4e420f0>
obj: <__main__.MyClass object at 0x7f19d4e42160>
objtype: <class '__main__.MyClass'>

EX2類屬性

如果通過類直接訪問屬性x,那么obj接直接為None,這還是比較好理解,因為不存在MyClass的實例。

>>> MyClass.x
self in RevealAccess: <__main__.RevealAccess object at 0x7f53651070f0>
self: <__main__.RevealAccess object at 0x7f53651070f0>
obj: None
objtype: <class '__main__.MyClass'>

描述符的原理

描述符觸發

上面這個例子中,我們分別從實例屬性和類屬性的角度列舉了描述符的用法,下面我們來仔細分析一下內部的原理:

  • 如果是對實例屬性進行訪問,實際上調用了基類object的__getattribute__方法,在這個方法中將obj.d轉譯成了type(obj).__dict__['d'].__get__(obj, type(obj))
  • 如果是對類屬性進行訪問,相當于調用了元類type的__getattribute__方法,它將cls.d轉譯成cls.__dict__['d'].__get__(None, cls),這里__get__()的obj為的None,因為不存在實例。

簡單講一下__getattribute__魔術方法,這個方法在我們訪問一個對象的屬性的時候會被無條件調用,詳細的細節比如和__getattr__getitem__的區別我會在文章的末尾做一個額外的補充,我們暫時并不深究。

描述符優先級

首先,描述符分為兩種:

  • 如果一個對象同時定義了__get__()和__set__()方法,則這個描述符被稱為data descriptor
  • 如果一個對象只定義了__get__()方法,則這個描述符被稱為non-data descriptor

我們對屬性進行訪問的時候存在下面四種情況:

  • data descriptor
  • instance dict
  • non-data descriptor
  • __getattr__()

它們的優先級大小是:

data descriptor > instance dict > non-data descriptor > __getattr__()

這是什么意思呢?就是說如果實例對象obj中出現了同名的data descriptor->d 和 instance attribute->dobj.d對屬性d進行訪問的時候,由于data descriptor具有更高的優先級,Python便會調用type(obj).__dict__['d'].__get__(obj, type(obj))而不是調用obj.__dict__[‘d’]。但是如果描述符是個non-data descriptor,Python則會調用obj.__dict__['d']

Property

每次使用描述符的時候都定義一個描述符類,這樣看起來非常繁瑣。Python提供了一種簡潔的方式用來向屬性添加數據描述符。

property(fget=None, fset=None, fdel=None, doc=None) -> property attribute

fget、fset和fdel分別是類的getter、setter和deleter方法。我們通過下面的一個示例來說明如何使用Property:

class Account(object):
    def __init__(self):
        self._acct_num = None
    def get_acct_num(self):
        return self._acct_num
    def set_acct_num(self, value):
        self._acct_num = value
    def del_acct_num(self):
        del self._acct_num
    acct_num = property(get_acct_num, set_acct_num, del_acct_num, '_acct_num property.')

如果acct是Account的一個實例,acct.acct_num將會調用getter,acct.acct_num = value將調用setter,del acct_num.acct_num將調用deleter。

>>> acct = Account()
>>> acct.acct_num = 1000
>>> acct.acct_num
1000

Python也提供了@property裝飾器,對于簡單的應用場景可以使用它來創建屬性。一個屬性對象擁有getter,setter和deleter裝飾器方法,可以使用它們通過對應的被裝飾函數的accessor函數創建屬性的拷貝。

class Account(object):
    def __init__(self):
        self._acct_num = None
    @property
     # the _acct_num property. the decorator creates a read-only property
    def acct_num(self):
        return self._acct_num
    @acct_num.setter
    # the _acct_num property setter makes the property writeable
    def set_acct_num(self, value):
        self._acct_num = value
    @acct_num.deleter
    def del_acct_num(self):
        del self._acct_num

如果想讓屬性只讀,只需要去掉setter方法。

在運行時創建描述符

我們可以在運行時添加property屬性:

class Person(object):
    def addProperty(self, attribute):
        # create local setter and getter with a particular attribute name
        getter = lambda self: self._getProperty(attribute)
        setter = lambda self, value: self._setProperty(attribute, value)
        # construct property attribute and add it to the class
        setattr(self.__class__, attribute, property(fget=getter, \
                                                    fset=setter, \
                                                    doc="Auto-generated method"))
    def _setProperty(self, attribute, value):
        print("Setting: {} = {}".format(attribute, value))
        setattr(self, '_' + attribute, value.title())
    def _getProperty(self, attribute):
        print("Getting: {}".format(attribute))
        return getattr(self, '_' + attribute)
>>> user = Person()
>>> user.addProperty('name')
>>> user.addProperty('phone')
>>> user.name = 'john smith'
Setting: name = john smith
>>> user.phone = '12345'
Setting: phone = 12345
>>> user.name
Getting: name
'John Smith'
>>> user.__dict__
{'_phone': '12345', '_name': 'John Smith'}

靜態方法和類方法

我們可以使用描述符來模擬Python中的@staticmethod@classmethod的實現。我們首先來瀏覽一下下面這張表:

Transformation Called from an Object Called from a Class
function f(obj, *args) f(*args)
staticmethod f(*args) f(*args)
classmethod f(type(obj), *args) f(klass, *args)

靜態方法

對于靜態方法fc.fC.f是等價的,都是直接查詢object.__getattribute__(c, ‘f’)或者object.__getattribute__(C, ’f‘)。靜態方法一個明顯的特征就是沒有self變量。

靜態方法有什么用呢?假設有一個處理專門數據的容器類,它提供了一些方法來求平均數,中位數等統計數據方式,這些方法都是要依賴于相應的數據的。但是類中可能還有一些方法,并不依賴這些數據,這個時候我們可以將這些方法聲明為靜態方法,同時這也可以提高代碼的可讀性。

使用非數據描述符來模擬一下靜態方法的實現:

class StaticMethod(object):
    def __init__(self, f):
        self.f = f
    def __get__(self, obj, objtype=None):
        return self.f

我們來應用一下:

class MyClass(object):
    @StaticMethod
    def get_x(x):
        return x
print(MyClass.get_x(100))  # output: 100

類方法

Python的@classmethod@staticmethod的用法有些類似,但是還是有些不同,當某些方法只需要得到類的引用而不關心類中的相應的數據的時候就需要使用classmethod了。

使用非數據描述符來模擬一下類方法的實現:

class ClassMethod(object):
    def __init__(self, f):
        self.f = f
    def __get__(self, obj, klass=None):
        if klass is None:
            klass = type(obj)
        def newfunc(*args):
            return self.f(klass, *args)
        return newfunc

其他的魔術方法

***接觸Python魔術方法的時候,我也被__get____getattribute____getattr____getitem__之間的區別困擾到了,它們都是和屬性訪問相關的魔術方法,其中重寫__getattr____getitem__來構造一個自己的集合類非常的常用,下面我們就通過一些例子來看一下它們的應用。

__getattr__

Python默認訪問類/實例的某個屬性都是通過__getattribute__來調用的,__getattribute__會被無條件調用,沒有找到的話就會調用__getattr__。如果我們要定制某個類,通常情況下我們不應該重寫__getattribute__,而是應該重寫__getattr__,很少看見重寫__getattribute__的情況。

從下面的輸出可以看出,當一個屬性通過__getattribute__無法找到的時候會調用__getattr__

In [1]: class Test(object):
    ...:     def __getattribute__(self, item):
    ...:         print('call __getattribute__')
    ...:         return super(Test, self).__getattribute__(item)
    ...:     def __getattr__(self, item):
    ...:         return 'call __getattr__'
    ...:
In [2]: Test().a
call __getattribute__
Out[2]: 'call __getattr__'

應用

對于默認的字典,Python只支持以obj['foo']形式來訪問,不支持obj.foo的形式,我們可以通過重寫__getattr__讓字典也支持obj['foo']的訪問形式,這是一個非常經典常用的用法:

class Storage(dict):
    """     A Storage object is like a dictionary except `obj.foo` can be used     in addition to `obj['foo']`.     """
    def __getattr__(self, key):
        try:
            return self[key]
        except KeyError as k:
            raise AttributeError(k)
    def __setattr__(self, key, value):
        self[key] = value
    def __delattr__(self, key):
        try:
            del self[key]
        except KeyError as k:
            raise AttributeError(k)
    def __repr__(self):
        return '<Storage ' + dict.__repr__(self) + '>'

我們來使用一下我們自定義的加強版字典:

>>> s = Storage(a=1)
>>> s['a']
1
>>> s.a
1
>>> s.a = 2
>>> s['a']
2
>>> del s.a
>>> s.a
...
AttributeError: 'a'

__getitem__

getitem用于通過下標[]的形式來獲取對象中的元素,下面我們通過重寫__getitem__來實現一個自己的list。

class MyList(object):
    def __init__(self, *args):
        self.numbers = args
    def __getitem__(self, item):
        return self.numbers[item]
my_list = MyList(1, 2, 3, 4, 6, 5, 3)
print my_list[2]

這個實現非常的簡陋,不支持slice和step等功能,請讀者自行改進,這里我就不重復了。

應用

下面是參考requests庫中對于__getitem__的一個使用,我們定制了一個忽略屬性大小寫的字典類。

程序有些復雜,我稍微解釋一下:由于這里比較簡單,沒有使用描述符的需求,所以使用了@property裝飾器來代替,lower_keys的功能是將實例字典中的鍵全部轉換成小寫并且存儲在字典self._lower_keys中。重寫了__getitem__方法,以后我們訪問某個屬性首先會將鍵轉換為小寫的方式,然后并不會直接訪問實例字典,而是會訪問字典self._lower_keys去查找。賦值/刪除操作的時候由于實例字典會進行變更,為了保持self._lower_keys和實例字典同步,首先清除self._lower_keys的內容,以后我們重新查找鍵的時候再調用__getitem__的時候會重新新建一個self._lower_keys

class CaseInsensitiveDict(dict):
    @property
    def lower_keys(self):
        if not hasattr(self, '_lower_keys') or not self._lower_keys:
            self._lower_keys = dict((k.lower(), k) for k in self.keys())
        return self._lower_keys
    def _clear_lower_keys(self):
        if hasattr(self, '_lower_keys'):
            self._lower_keys.clear()
    def __contains__(self, key):
        return key.lower() in self.lower_keys
    def __getitem__(self, key):
        if key in self:
            return dict.__getitem__(self, self.lower_keys[key.lower()])
    def __setitem__(self, key, value):
        dict.__setitem__(self, key, value)
        self._clear_lower_keys()
    def __delitem__(self, key):
        dict.__delitem__(self, key)
        self._lower_keys.clear()
    def get(self, key, default=None):
        if key in self:
            return self[key]
        else:
            return default

我們來調用一下這個類:

>>> d = CaseInsensitiveDict()
>>> d['ziwenxie'] = 'ziwenxie'
>>> d['ZiWenXie'] = 'ZiWenXie'
>>> print(d)
{'ZiWenXie': 'ziwenxie', 'ziwenxie': 'ziwenxie'}
>>> print(d['ziwenxie'])
ziwenxie
# d['ZiWenXie'] => d['ziwenxie']
>>> print(d['ZiWenXie'])
ziwenxie
責任編輯:張燕妮 來源: ZiWenXie
相關推薦

2012-08-08 10:31:41

IBMdW

2016-10-28 21:55:28

Javascript屬性特性屬性描述符

2025-01-10 15:13:38

2009-07-08 09:46:45

Servlet注釋部署描述符

2023-04-06 15:22:15

Linux進程系統

2022-05-20 12:40:23

PythonMetaclass

2016-10-19 15:15:26

2019-03-05 22:15:08

BashLinux命令

2019-07-05 14:20:45

RPC服務器模型

2020-04-03 13:43:23

Python列表推導式字典推導式

2009-09-04 14:04:53

C#文檔

2020-02-07 18:16:01

進程線程底層原理

2023-12-13 14:01:34

Elasticsea文件描述符操作系統

2021-06-18 06:02:24

內核文件傳遞

2023-12-25 14:50:39

Python迭代器

2021-05-19 14:48:58

Linux文件fd

2025-04-09 11:20:00

LINQ代碼數據處理

2020-04-10 09:55:28

Git 工具黑魔法

2019-07-09 15:30:31

Linuxulimit文件描述符

2019-07-09 14:30:16

LinuxJava 服務器
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 人人干人人草 | 亚洲v日韩v综合v精品v | 免费一区二区 | 黄网站在线观看 | 中文字字幕在线中文乱码范文 | 国产精品毛片无码 | 超碰成人免费 | 91国内在线观看 | 久久精品国产免费一区二区三区 | 在线国产一区 | 最新av在线网址 | a国产一区二区免费入口 | 欧美色视频免费 | 亚洲二区在线观看 | 国产伦精品一区二区三区高清 | 91免费视频 | 欧美日韩精品久久久免费观看 | 一区二区三区视频在线免费观看 | 91色网站 | 99免费在线观看 | 一区二区三区av | 日韩精品成人一区二区三区视频 | 国产乱码精品1区2区3区 | 亚洲欧美激情精品一区二区 | 天天视频成人 | 色资源在线视频 | 日本亚洲欧美 | 精品欧美一区二区在线观看视频 | 久久精品亚洲一区二区三区浴池 | 久久成人免费观看 | 亚洲综合色网站 | 91精品一区二区三区久久久久 | 国产乱码精品1区2区3区 | 天堂在线免费视频 | 午夜看电影在线观看 | 精品视频www | 日本成年免费网站 | 日本精品视频在线 | 黄色毛片在线观看 | 小草久久久久久久久爱六 | 婷婷久久五月天 |