python 描述器

描述器定义

① 实现描述符协议
  • 实现 __get__(), __set__(), __delete__() 方法
② 实例&类属性、方法的访问
  • 1、类和实例的访问
    # 标准的描述符定义
    class Quanty:
        def __init__(self, weight):
            self.weight = weight
        
        def __get__(self, instance, cls):
            return self.weight
            
        def __set__(self, instance, value):
            self.weight = value
    
    class Person:
        name = 'PY_ONE'
        weight = Quanty(90)  # 描述器实例
        
        def __init__(self, age):
            self.age = age
        
        @classmethod
        def get_verson(cls):
            return 1.0
        
        @staticmethod
        def find_version():
            return 1.0
    
    • 使用 vars()__dict__查看对象的属性
    # Person 类对象
    >>> vars(Person)
    >>> mappingproxy({
        '__dict__': <attribute '__dict__' of 'Person' objects>,
        '__doc__': None,
        '__init__': <function __main__.Person.__init__>,
        '__module__': '__main__',
        '__weakref__': <attribute '__weakref__' of 'Person' objects>,
        'find_version': <staticmethod at 0x10544ed30>,
        'get_verson': <classmethod at 0x10544e240>,
        'name': 'PY_ONE',
        'weight': <__main__.Quanty at 0x105472cc0>
    })
    
    # Person 实例对象
    >>> per = Person(18)
    >>> vars(per)
    >>> {'age': 18}
    
    # 可以看到,实例对象 per 只有 age 一个属性,类 Person 有 name, get_version, find_version, weight 属性
    
    • 类属性可以使用实例或类对象访问,只有类才能修改
    >>> p1, p2 = Person(16), Person(17)
    >>> p1.name  # PY_ONE
    >>> p2.name  # PY_ONE
    >>> Person.name  # PY_ONE
    
    >>> p1.name = 'PY_WJ'
    >>> Person.name  # PY_ONE
    >>> p2.name  # PY_ONE
    >>> p1.name  # PY_ONE
    
    >>> Person.name = '_PY'
    >>> p2.name  # _PY
    >>> p1.name  # PY_WJ
    
    • 属性访问的原理与描述器
    # 属性访问就是基于 Python 的描述器实现。类或者实例通过 . 操作符访问属性,会先访问对象的 __dict__,如果没有再访问类(或父类,元类除外)的 __dict__。如果最后这个 __dict__ 的对象是一个描述器,则会调用描述器的 __get__ 方法
    
    >>> p2.name  # 等同于调用 p2.__dict__,发现 name 不在此属性字典中,则尝试调用 type(p2).__dict__,发现此属性存在,且此属性的对象是字符串,故为 type(p2).__dict__['name']
    
    >>> p2.weight  # 和上述调用一致,不过在 type(p2).__dict__['weight']的对象是一个描述器,故最终调用如下 type(p2).__dict__['weight'].__get__(p2, type(p2))
    
    • 类方法的调用
    >>> p1.get_version  # 调用同上,发现 p1 实例中没有 get_version 属性,且 type(p1).__dict__['get_version'] 的对象是 classmethod 实例,故最终调用如下 type(p1).__dict__['get_version'].__get__(Person)
    
    • 静态方法同类方法
    # 同类方法的调用
    
  • 2、使用描述符对类属性做验证
    • 示例如下
    # 验证正数的描述器
    class PositiveNum:
        
        def __init__(self, key_col):
            self._key_col = key_col
        
        def __get__(self, instance, cls):
            return instance.__dict__[self._key_col]
            
        def __set__(self, instance, value):
            if value < 0:
                raise ValueError('value must be > 0')
            instance.__dict__[self._key_col] = value
    
    # 商品类
    class Good:
        price = PositiveNum('price')
        quantity = PositiveNum('quantity')
        
        def __init__(self, price, quantity):
            self.price = price
            self.quantity = quantity
    
    >>> g = Good(12, -1)  # ValueError
    >>> g = Good(12, 8)  # {'price': 12, 'quantity': 8}
    
    # 疑问点 a
    >>> g.price = -1  # ValueError:  会执行type(g).__dict__['price'].__set__(g, -1)
    
    # 疑问点 b
    >>> Good.price = -1  # TODO 
    
  • 3、疑问点解释
    • 疑问点 a
    # 只是对类属性声明为描述符对象,为何对实例属性进行赋值操作的时候,也会被描述符覆盖?
    
    答: 
    1、描述符分为覆盖型描述符和非覆盖型描述符,而判断是否为覆盖型描述符的条件就是该描述符是否实现了 __set__(self, obj, value) 方法。
    2、对于同名实例属性,如上文的 price 和 quantity 属性,如果描述符为覆盖型描述符,则描述符同样会覆盖对实例属性的复制操作。
    3、当描述符和实例字典中的某个属性重名,按访问优先级为 覆盖型描述符 > 同名实例字典中属性 > 非覆盖型描述符
    
    • 疑问点 b
    # 类属性已声明为描述符对象,为何对类属性赋值会覆盖描述符?
    
    答:
    1、不管描述符是不是覆盖型描述符,对类属性赋值都能覆盖描述符
    2、这是一种猴子补丁技术
    
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,123评论 6 490
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,031评论 2 384
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 156,723评论 0 345
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,357评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,412评论 5 384
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,760评论 1 289
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,904评论 3 405
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,672评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,118评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,456评论 2 325
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,599评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,264评论 4 328
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,857评论 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,731评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,956评论 1 264
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,286评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,465评论 2 348

推荐阅读更多精彩内容

  • 1.1. 摘要 定义描述器, 总结描述器协议,并展示描述器是怎么被调用的。展示一个自定义的描述器和包括函数,属性(...
    mutex73阅读 444评论 0 2
  • 描述器协议 描述器协议包括以下3个方法: object.__get__(self, instance, owner...
    王吉吉real阅读 833评论 0 0
  • 语法简析 一般来说,描述器(descriptor)是一个有”绑定行为”的对象属性(object attribute...
    ButteredCat阅读 4,068评论 1 4
  • Python 面向对象Python从设计之初就已经是一门面向对象的语言,正因为如此,在Python中创建一个类和对...
    顺毛阅读 4,211评论 4 16
  • 第1章 醒悟 1-孰主孰仆 在这个章节中,作者用美国数学家约翰·纳什“用自己的精神战胜了自己的精神病”,以及奥地利...
    夏花争妍阅读 162评论 0 0