zoukankan      html  css  js  c++  java
  • Python的__getattr__和__getattribute__

    __getattr__
    __getattr__在当前主流的Python版本中都可用,重载__getattr__方法对类及其实例未定义的属性有效。也就属性是说,如果访问的属性存在,就不会调用__getattr__方法。这个属性的存在,包括类属性和实例属性。

    Python官方文档的定义

    Called when an attribute lookup has not found the attribute in the usual places (i.e. it is not an instance attribute nor is it found in the class tree for self). name is the attribute name.

    class ClassA:
    
        x = 'a'
    
        def __init__(self):
            self.y = 'b'
    
        def __getattr__(self, item):
            return '__getattr__'
    
    if __name__ == '__main__':
        a = ClassA()
        # 输出结果 a
        print(a.x)
        # 使用实例直接访问实例存在的实例属性时,不会调用__getattr__方法
        # 输出结果 b
        print(a.y)
        # 使用实例直接访问实例不存在的实例属性时,会调用__getattr__方法
        # 输出结果 __getattr__
        print(a.z)

    __getattribute__
    __getattribute__仅在新式类中可用,重载__getattrbute__方法对类实例的每个属性访问都有效。

    Python官方文档的定义

    Called unconditionally to implement attribute accesses for instances of the class.

    示例代码:

    class ClassA:
    
        x = 'a'
    
        def __init__(self):
            self.y = 'b'
    
        def __getattribute__(self, item):
            return '__getattribute__'
    
    
    if __name__ == '__main__':
        a = ClassA()
        # 使用实例直接访问存在的类属性时,会调用__getattribute__方法
        # 输出结果 __getattribute__
        print(a.x)
        # 使用实例直接访问实例存在的实例属性时,会调用__getattribute__方法
        # 输出结果 __getattribute__
        print(a.y)
        # 使用实例直接访问实例不存在的实例属性时,也会调用__getattribute__方法
        # 输出结果 __getattribute__
        print(a.z)

    运行结果:

    __getattribute__
    __getattribute__
    __getattribute__

    另外,当同时定义__getattribute__和__getattr__时,__getattr__方法不会再被调用,除非显示调用__getattr__方法或引发AttributeError异常。

    示例代码(__getattr__方法不会再被调用):

    class ClassA:
    
        def __getattr__(self, item):
            print('__getattr__')
    
        def __getattribute__(self, item):
            print('__getatttribute__')
    
    if __name__ == '__main__':
        a = ClassA()
        a.x

    运行结果:

    __getatttribute__

    由于__getattr__只针对未定义属性的调用,所以它可以在自己的代码中自由地获取其他属性,而__getattribute__针对所有的属性运行,因此要十分注意避免在访问其他属性时,再次调用自身的递归循环。

    当在__getattribute__代码块中,再次执行属性的获取操作时,会再次触发__getattribute__方法的调用,代码将会陷入无限递归,直到Python递归深度限制(重载__setter__方法也会有这个问题)。

    示例代码(无限递归):

    class ClassA:
    
        x = 'a'
    
        def __getattribute__(self, item):
            print('__getattribute__')
            return self.item
    
    
    if __name__ == '__main__':
        a = ClassA()
        a.x

    运行结果引发异常,提示达到最大递归深度

    ecursionError: maximum recursion depth exceeded

    同时,也没办法通过从__dict__取值的方式来避免无限递归

    class ClassA:
    
        x = 'a'
    
        def __getattribute__(self, name):
            return self.__dict__[name]
    
    
    if __name__ == '__main__':
        a = ClassA()
        # 无限递归
        a.x

    为了避免无限递归,应该把获取属性的方法指向一个更高的超类,例如object(因为__getattribute__只在新式类中可用,而新式类所有的类都显式或隐式地继承自object,所以对于新式类来说,object是所有新式类的超类)。

    修改代码(避免无限递归循环):

    class ClassA:
    
        x = 'a'
    
        def __getattribute__(self, item):
            print('__getattribute__')
            return super().__getattribute__(self, item)
    
    
    if __name__ == '__main__':
        a = ClassA()
        print(a.x)

    运行结果正常:

    __getattribute__
    a

    参考资料:

    https://docs.python.org/3/reference/datamodel.html

  • 相关阅读:
    redis的两种备份方式
    Vue—事件修饰符
    css3实现颤动的动画
    初学者可能不知道的vue技巧
    使用slot-scope复制vue中slot内容
    pre-commit钩子,代码质量检查
    爬虫可视化点选配置工具之获取鼠标点选元素
    Vue源码探究-事件系统
    使用electron实现百度网盘悬浮窗口功能!
    electron实现qq快捷登录!
  • 原文地址:https://www.cnblogs.com/blackmatrix/p/5681480.html
Copyright © 2011-2022 走看看