鱼C论坛

 找回密码
 立即注册
查看: 948|回复: 4

[已解决]python第45课最后一题求助

[复制链接]
发表于 2022-1-19 20:34:26 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

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了呢??? 求教各位大佬!
最佳答案
2022-1-20 15:37:39
本帖最后由 阿奇_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类的实例对象。。 然后这个对象有什么特点和作用呢?……


  你自己琢磨吧,吾已猝~



想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复

使用道具 举报

发表于 2022-1-19 21:50:19 | 显示全部楼层
super().__setattr__('counter',self.counter+1) #那么这里就是让父类的实例化对象的属性counter为1

这里是加一哦,不知道你注释是不是写错了
为什么 c.counter也为1了呢

你是说这里的吗
c = Counter()
c.x = 1
c.counter
1
c.x =1 不是执行了__setattr__方法吗?又执行了super().__setattr__('counter',self.counter+1) 方法,那不是0 + 1 =1 吗
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2022-1-20 15:37:39 | 显示全部楼层    本楼为最佳答案   
本帖最后由 阿奇_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类的实例对象。。 然后这个对象有什么特点和作用呢?……


  你自己琢磨吧,吾已猝~



想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2022-1-20 18:46:33 | 显示全部楼层
大马强 发表于 2022-1-19 21:50
这里是加一哦,不知道你注释是不是写错了

你是说这里的吗

还是非常感谢您的回答,我昨天晚上又想了半天想出来了。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2022-1-20 18:48:38 | 显示全部楼层
阿奇_o 发表于 2022-1-20 15:37
虽然super的原理,一下子不好完全理解,但你这里 可以看看下面的代码,可能 "暂时就明白了":

谢谢您的回答,我发到论坛之后又琢磨了一些实例,发现确实是我大方向搞错了,小甲鱼的例子其实全是关于self这个实例对象本身的,跟父类的实例对象没有关系,这里就像您说的只是运用了父类的方法,具体方法是什么课上没说,但是比较容易猜出来,就是字面意思。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-1-12 09:56

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表