Python语言学习讲解十七:python之__gettattr__、__getattribute__和__setattr__详解

 

 

注:每周一到周五都会进行相关Python基础知识更新,欢迎大家提宝贵的意见

 

python语言提供了一种挂钩,使得开发者能够很方便的编写出通用代码,他们使用的不是普通实例的属性,@property方法和描述符,而是使用的是python的魔术方法

__gettattr__、__getattribute__和__setattr__,他们属于动态行为。

>>> class TEST(object):

def __init__(self):
self.value = 1
def __getattr__(self, name):
value = 'Value for %s' % name
setattr(self, name, value)
return value
>>> testobj = TEST()
>>> print('---:',testobj.__dict__)
{'__methods__': 'Value for __methods__', '__members__': 'Value for __members__', 'value': 1}

>>> print(testobj.__dict__)
{'__methods__': 'Value for __methods__', '__members__': 'Value for __members__', 'value': 1}
>>> print(testobj.attr)
'Value for attr'
>>> print(testobj.__dict__)
{'__methods__': 'Value for __methods__', '__members__': 'Value for __members__', 'value': 1, 'attr': 'Value for attr'}

 

总结:

__gettattr__:如果某个类定义了这个方法,并且在该类的对象的字典中又找不到相应的属性时候,那么次方法会被调用。

__getattribute__:不管对象的字典中有没有找到对应的属性,都会调用

__setattr__:无论是直接赋值还是通过内置的setattr函数赋值,都会调用

********还有一点需要住的是__getattribute__和__setattr__方法中访问实例属性的时候,应该直接通过super()来做,避免无线递归。******

 

# -*- coding: utf-8 -*-

class C(object):
    a = 'abc'

    def __getattribute__(self, name):
        print("__getattribute__() is called name =", name)
        return object.__getattribute__(self, name)

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

    def __get__(self, instance, owner):
        print("__get__() is called instance = {} owner = {}".format(instance, owner))
        return self

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


class C2(object):
    mc = C()


if __name__ == '__main__':
    c = C()
    c2 = C2()
    c.a
    print('-----------存在的属性:__getattribute__------------\n')
    c.noattr
    print('-----------不存在的属性__getattribute__ ---》 __getattr__------------\n')
    C2.mc
    print('----------类直接访问成员(实现了__get__的类)都会先经过__get__函数-------------\n')
    c2.mc
    print('-----------对象直接访问成员(实现了__get__的类)都会先经过__get__函数------------\n')

    C2.mc.a
    print('----------类把直接访问成员(实现了__get__的类)中存在的属性:__get__ --> __getattribute__  -------------\n')
    c2.mc.b
    print('-----------对象直接访问成员(实现了__get__的类)中不存在的属性:__get__ --> __getattribute__ -->__getattr__ ------------\n')
    # print(c2.d.a)

 

你可能感兴趣的:(python基础)