python中__get__,__getattr__,__getattribute__的区别

__get__,__getattr__和__getattribute都是访问属性的方法,但不太相同。  object.__getattr__(self, name)  当一般位置找不到attribute的时候,会调用getattr,返回一个值或AttributeError异常。  object.__getattribute__(self, name)  无条件被调用,通过实例访问属性。如果class中定义了__getattr__(),则__getattr__()不会被调用(除非显示调用或引发AttributeError异常)  object.__get__(self, instance, owner)  如果class定义了它,则这个class就可以称为descriptor。owner是所有者的类,instance是访问descriptor的实例,如果不是通过实例访问,而是通过类访问的话,instance则为None。(descriptor的实例自己访问自己是不会触发__get__,而会触发__call__,只有descriptor作为其它类的属性才有意义。)(所以下文的d是作为C2的一个属性被调用) 

class C(object):
    a = 'abc'def __getattribute__(self, name):
        print("__getattribute__() is called"),name
        return object.__getattribute__(self, name)

    def __getattr__(self, name):
        print("__getattr__() is called "),name
        return name + " from getattr"

    def __get__(self, instance, owner):
        print("__get__() is called", instance, owner)
        return self

    def foo(self, x):
        print(x)


class C2(object):
    d = C()


if __name__ == '__main__':
    c = C()
    c2 = C2()
    print(c.a)
    print(c.zzzzzzzz)
    print '==============1'
    C2.d
    print '==============2'
    print(c2.d.a)

 结果

__getattribute__() is called a
abc
__getattribute__() is called zzzzzzzz
__getattr__() is called  zzzzzzzz
zzzzzzzz from getattr
==============1
('__get__() is called', None, <class '__main__.C2'>)
==============2
('__get__() is called', <__main__.C2 object at 0x00000000030BCDA0>, <class '__main__.C2'>)
__getattribute__() is called a
abc

小结:可以看出,每次通过实例访问属性,都会经过__getattribute__函数。而当属性不存在时,仍然需要访问__getattribute__,不过接着要访问__getattr__。这就好像是一个异常处理函数。  每次访问descriptor(即实现了__get__的类),都会先经过__get__函数。  需要注意的是,当使用类访问不存在的变量是,不会经过__getattr__函数。而descriptor不存在此问题,只是把instance标识为none而已。

参考:http://luozhaoyu.iteye.com/blog/1506426

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Laoqi's Linux运维专列

shell中的函数+数组+数组分片

43410
来自专栏北京马哥教育

百度大牛总结十条Python面试题陷阱,看看你是否会中招

2207
来自专栏mukekeheart的iOS之旅

《JavaScript高级程序设计》学习笔记(3)——变量、作用域和内存问题

欢迎关注本人的微信公众号“前端小填填”,专注前端技术的基础和项目开发的学习。 本节内容对应《JavaScript高级程序设计》的第四章内容。 1、函数:通过函数...

2906
来自专栏cs

c++那些事儿6.0 STL中的string

---- 知识点综述 c++,string 类 string 是C++中的字符串对象,是一种特殊类型的容器,专门设计来操作的字符序列。 ...

2877
来自专栏Java编程

Java虚拟机体系结构

一个运行时的Java虚拟机实例的天职是:负责运行一个java程序。当启动一个Java程序时,一个虚拟机实例也就诞生了。当该程序关闭退出,这个虚拟机实例也就随之消...

7491
来自专栏搞前端的李蚊子

使用箭头函数的几个注意事项

刚才写一个需要递归操作的函数,在使用arguments.callee的时候,报错undefined,因为arguments.callee在严格模式下会失效,以为...

3756
来自专栏用户3030674的专栏

java单例模式

单例设计模式:解决一个类在内存中只存在一个对象  多用于环境变量设置等  单例模式的要求:1.只能有一个对象,禁止其他程序建立该类对象          2....

751
来自专栏cloudskyme

未绑定元素“mx:Panel”的前缀“mx”

描述 资源 路径 位置 类型 未绑定元素“mx:Panel”的前缀“mx”。 HelloWorld.mxml /HelloWorld/src 第 23 行 Fl...

2897
来自专栏黑泽君的专栏

打印println()方法的逻辑

782
来自专栏专注 Java 基础分享

Java并发之synchronized关键字

     上篇文章我们主要介绍了并发的基本思想以及线程的基本知识,通过多线程我们可以实现对计算机资源的充分利用,但是在最后我们也说明了多线程给程序带来的两种典型...

1995

扫码关注云+社区