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)

 

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

隨意的風

如果你觉得有帮助,期待你的打赏

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值