|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
x
感觉小甲鱼上课讲的不是很清楚,于是自己课下琢磨了一下。非常欢迎各位大佬指正!
class C: #这个例子中要特别注意所有的属性都是针对实例对象的。
def __getattribute__(self,name): #注意self就是指当前的C类的实例对象。此函数默认会返回实例对象中属性name的值!
print('getattribute')
return super().__getattribute__(name) # 注意super()就是表示父类的实例对象,这里由于有底下__setattr__的存在,导致父类object有一个实例对象的属性和值已被创建!
def __getattr__(self,name): #当self中的属性name不存在时自动调用,且无返回值。
print('getattr')
def __setattr__(self,name,value): #当设置self的属性时自动调用。
print('setattr')
super().__setattr__(name,value) #同时__setattr__函数还有为对象添加属性和值的功能,这里就为父类的某实例对象添加了属性name。
def __delattr__(self,name): #当删除self的属性时自动调用。
print('delattr')
super().__delattr__(name) #同时__delattr__函数还有为对象删除属性的功能,这里就删除了父类的某实例对象(同上)的属性。
不知道我这么理解对不对,如果没问题的话。动动手最后一题我就不能理解。
题目:编写一个Counter类,用于实时检测测对象有多少个属性。
程序实现如下:
c = Counter()
c.x = 1
c.counter
1
c.y =1
c.x =1
c.counter
3
del c.x
c.counter
2
答案如下:
class Counter:
def __init__(self):
super().__setattr__('counter',0) #这里我明白是实例化c时调用父类__setattr__让父类实例化对象的属性counter 为0.
def __setattr__(self,name,value):
super().__setattr__('counter',self.counter+1) #那么这里就是让父类的实例化对象的属性counter为1
super().__setattr__(name,value)
def __delattr__(self,name):
super().__setattr__('counter',self.counter-1)
super().__delattr__(name)
所以根据我的个人理解应该时父类的实例化对象 也就是 super()的属性counter为1呀,c不是Counter的实例化对象吗?为什么 c.counter也为1了呢??? 求教各位大佬!
本帖最后由 阿奇_o 于 2022-1-20 18:26 编辑
虽然super的原理,一下子不好完全理解,但你这里 可以看看下面的代码,可能 "暂时就明白了": >>> class Counter:
def __init__(self):
super().__setattr__('counter', 0)
>>> c = Counter()
>>> c.counter
0
>>> c.counter = c.counter + 1
>>> c.counter
1
>>> c.__dir__()
['counter', '__module__', '__init__', '__dict__', '__weakref__', '__doc__', '__repr__', '__hash__', '__str__', '__getattribute__', '__setattr__', '__delattr__', '__lt__', '__le__', '__eq__', '__ne__', '__gt__', '__ge__', '__new__', '__reduce_ex__', '__reduce__', '__subclasshook__', '__init_subclass__', '__format__', '__sizeof__', '__dir__', '__class__']
>>> c.__setattr__('x', 1)
>>> c.x
1
>>> c.__setattr__('y', 1)
>>> c.__dir__()
['counter', 'x', 'y', .....省略 ]
其实一开始的 super 也可以改写成下面这样"更常见的"方式:
>>> class Counter:
def __init__(self):
self.__setattr__('counter', 0)
>>> c = Counter()
>>> c.counter
0
>>> class Counter:
def __init__(self):
#self.__setattr__('counter', 0)
self.counter = 0 # 这很常见了吧
补充:我又琢磨了许久,你的例子,其实是你用了super()也根本没用到"父类的实例",而是用了 super类的对象的 __setattr__方法,因为
这里涉及了 __setattr__这样的"特殊方法",它的作用就是"动态修改 本类的属性状态",而不是"父类"的。
所以,当你 c=Counter(); super(Counter, c).__setattr__('x', 1) 它就是在本类Counter里动态地增加了 x 这个属性,于是 c.x 就能读取到。
为什么说 用的是 super类的对象 呢? 因为 super()或super(Counter, self)返回的就是这样一个 super类的实例对象。。 然后这个对象有什么特点和作用呢?……
你自己琢磨吧,吾已猝~
|
|