这两个方法有太多的相似度,以至于很长一段时间把它们混为一谈,其实是不一样的,__setattr__是对类方法的重写,我们一般很少用。下面操作就会报错。
class Foo:
def __init__(self):
self.__data = {}
def __setitem__(self, key, value): # 对象像字典那样操作
self.__data[key] = value
def __setattr__(self, key, value): # 方法重写
print(key, value)
上面原因为在创建类时,我写了__setattr__(self,key,value)方法,我写了之后,根本就没做任何事情,顺便把父类的方法给覆盖了,当程序要执行__setitem__方法时,它要先调用__setattr__做事情(实际是把属性写进__dict__里面)。解决办法就是把父类方法重写一遍。
class Foo:
def __init__(self):
self.__data = {}
def __setitem__(self, key, value): # 对象像字典那样操作
self.__data[key] = value
def __setattr__(self, key, value): # 方法重写
print(key, value)
super().__setattr__(key,value)
class Foo:
def __init__(self):
self.__data = {}
def __setitem__(self, key, value): # 对象像字典那样操作
self.__data[key] = value
def __getitem__(self, item): return self.__data[item]
def __delitem__(self, key): del self.__data[key]
def __getattr__(self, item):
print(item)
def __setattr__(self, key, value): # 方法重写
print(key, value)
super().__setattr__(key, value)
def __delattr__(self, item): # 方法重写
print(item)
super().__delattr__(item)
f = Foo()
f['name'] = 'xuan'
f['age'] = 8
f['hobby'] = 'eatting'
f.m = 'hello'
f.school = 'jiue'
print(f.__dict__)