python中的__dict__,__getattr__,__setattr__

python class 通过内置成员dict 存储成员信息(字典)

首先用一个简单的例子看一下dict 的用法

class A():
    def __init__(self,a,b):
        self.a = a
        self.b = b
    def f(self):
        print (self.__dict__)
a = A(1,2)
a.f()
输出结果:{‘b’: 2, ‘a’: 1}

我们可以通过重载getattr和setattr来拦截对成员的访问或者作出一些自己希望的行为
getattr 在访问对象访问类中不存在的成员时会自动调用

'''
遇到问题没人解答?小编创建了一个Python学习交流QQ群:857662006 
寻找有志同道合的小伙伴,互帮互助,群里还有不错的视频学习教程和PDF电子书!
'''
class A():
    def __init__(self,a,b):
        self.a = a
        self.b = b
    def f(self):
        print (self.__dict__)
    def __getattr__(self,name):
        print ("__getattr__")
a = A(1,2)
a.f()
a.x

setattr 方法

class A(object):
    def __init__(self,a,b):
        print('start init')
        self.a=a  #赋值语句会被__setattr__拦截
        self.b=b
        print('end init')
    def f(self):
        print(self.__dict__)#因为赋值语句被拦截,所以不能将内置成员存储成(字典)
    def __getattr__(self, name):
        print('__getattr__')
    def __setattr__(self, name,value):
        print('1__setattr__')
        self.__dict__[name]=value
        print('2__setattr__')
a=A(2,3)
print('---------')
a.aa=33

运行结果

如下:

start init
1__setattr__
2__setattr__
1__setattr__
2__setattr__
end init
---------
1__setattr__
2__setattr__
__setattr__

会拦截所有属性的的赋值语句。如果定义了这个方法,self.arrt = value 就会变成self,__ setattr__(“attr”, value).这个需要注意。

当在__ setattr__方法内对属性进行赋值是,不可使用self.attr = value,因为他会再次调用self,__ setattr __(“attr”, value),则会形成无穷递归循环,
最后导致堆栈溢出异常。应该通过对属性字典做索引运算来赋值任何实例属性,也就是使用self.dict[‘name’] = value

如:j.name=5 就会调用__ setattr__方法 self.[name]=5
因为这个类是从dict继承来的,是dict的超类
所以 self[attr]=value 相当于调用dict的下标方法
与 a={} ; a[attr]=value意思一样

注意:是拦截实例的赋值,类的赋值不能拦截

在提供的代码,`__setattr__`方法被重写为`dict.__setitem__`,这意味着当我们给对象的属性赋值时,实际上是调用了`dict.__setitem__`方法来设置字典的键值对。这样做的目的是为了让对象支持通过点操作符来设置属性,使得对象可以像访问属性一样访问字典的键值对。这种做法在某些情况下可以提供更直观和方便的使用方式。 除了`__setattr__`方法外,还有`__getattr__`和`__delattr__`方法。`__getattr__`方法被重写为`dict.get`方法,当属性不存在时,会返回None,而不是抛出AttributeError异常。`__setattr__`方法被重写为`dict.__setitem__`,用于设置属性值。`__delattr__`方法被重写为`dict.__delitem__`,用于删除属性。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* [python对象和字典dict的相互转化【点.和[属性]】的访问区别及特殊方法名总结](https://blog.csdn.net/weixin_43343144/article/details/92764884)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *2* [Python字典转对象的两种方法(dict -> object使属性用点来调用)](https://blog.csdn.net/qq_21567385/article/details/107990864)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] - *3* [python内置函数__getattr__、__serattr__、__delattr__与字典dict的使用](https://blog.csdn.net/SL1029_/article/details/129638430)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v92^chatsearchT3_1"}}] [.reference_item style="max-width: 33.333333333333336%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值