鱼C论坛

 找回密码
 立即注册
查看: 1821|回复: 5

[已解决]关于魔法方法__setattr__的疑问

[复制链接]
发表于 2020-7-17 23:50:38 | 显示全部楼层 |阅读模式

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

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

x
正常编写的代码:
class Rectangle:
    def __init__(self,width=0,height=0):
        self.square=width
        #self.height=height

    def __setattr__(self, name, value):

        if name=='square':
            self.width=value
            self.height=value
        else:
            #self.name=value
            super().__setattr__(name,value)

    def GetArra(self):
        return self.width*self.height

输入:r=Rectangle()能正常进行对象实例化,输入r.square=10 也能给属性正常赋值,不会进入无限回调。

改写__setattr__()后:
    def __setattr__(self, name, value):

        if name=='square':
            self.width=value
            self.height=value
        else:
            self.name=value
            #super().__setattr__(name,value)
        QQ截图20200717232827.png
输入:r=Rectangle()进入无限回调死循环。



问题1:在改写__setattr__()之前,执行r.square=10语句时,也会调用__setattr__()。在__setattr__()函数体中,执行self.width=value,此时应该等同于改写后执行 self.name=value语句,为何改写前不会进入死循环。

问题2:我猜想是因为基类的__setattr__()方法内部执行的原因,几番尝试后发现只有在__init__()函数体中的改变属性值才会触发无限回调,想了解基类__setattr__()方法的实现原理, 但是在pycharm中按F4只能看到一个pass,看不到函数体,请问基类__setattr__()方法的实现原理是怎样的?

问题3:我应该如何才能看到魔法方法的实现代码?最好是在pycharm中查看

问题有些多,请大佬们多些耐心


最佳答案
2020-7-18 07:45:41
本帖最后由 Twilight6 于 2020-7-18 07:49 编辑

问题1:在改写__setattr__()之前,执行r.square=10语句时,也会调用__setattr__()。
在__setattr__()函数体中,执行self.width=value,此时应该等同于改写后执行 self.name=value语句,为何改写前不会进入死循环。


第一个代码的 else 和 第二个代码的 else 不同。你的第一个代码只要变量名不为 square 时候

就执行 else 代码块,而 else 里面是只有调用父类方法来赋值这个代码,所以运行到这里就不会调用自身__setattr__ 方法了

而 self.width = value 只会调用自身__setattr__一次,因为 self.width 调用__setattr__的时候,name 是为 width ,不为 square 所以执行的是 else ,而 else 只有 super().__setattr__ 调用父类方法,所以就不会无限递归

而第二个代码,你不管执行 else 还是 if 代码块都只有赋值操作,就会无限调用自身

就好如你赋值调用了 __setattr__ , 结果进入 __setattr__ 方法后还是赋值操作,此时又调用自身,然后又是赋值操作又调用自身.....就永远没有出口了

而 第一个代码 设置了 else 中只有 super().__setattr__ 方法,你可以尝试着在 else 的代码块下 super() 魔法方法前面加个赋值操作,会发现又会是无限递归~


问题2、问题3:


去看源码大部分都是 C 语言的,详见 2 楼

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

使用道具 举报

发表于 2020-7-17 23:53:24 | 显示全部楼层
问题2:我猜想是因为基类的__setattr__()方法内部执行的原因,几番尝试后发现只有在__init__()函数体中的改变属性值才会触发无限回调,想了解基类__setattr__()方法的实现原理, 但是在pycharm中按F4只能看到一个pass,看不到函数体,请问基类__setattr__()方法的实现原理是怎样的?

问题3:我应该如何才能看到魔法方法的实现代码?最好是在pycharm中查看
https://github.com/python/cpython
原理就是 魔法(C语言)
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-7-17 23:57:48 | 显示全部楼层
问题1:在改写__setattr__()之前,执行r.square=10语句时,也会调用__setattr__()。在__setattr__()函数体中,执行self.width=value,此时应该等同于改写后执行 self.name=value语句,为何改写前不会进入死循环。
因为 self.name=value 还是需要调用 __setattr__ 自己。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-7-18 00:04:10 | 显示全部楼层
永恒的蓝色梦想 发表于 2020-7-17 23:57
因为 self.name=value 还是需要调用 __setattr__ 自己。

为什么self.width=value 没有调用__setattr__ 自己呢?这里不是也是改变了属性的值吗?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-7-18 07:45:41 | 显示全部楼层    本楼为最佳答案   
本帖最后由 Twilight6 于 2020-7-18 07:49 编辑

问题1:在改写__setattr__()之前,执行r.square=10语句时,也会调用__setattr__()。
在__setattr__()函数体中,执行self.width=value,此时应该等同于改写后执行 self.name=value语句,为何改写前不会进入死循环。


第一个代码的 else 和 第二个代码的 else 不同。你的第一个代码只要变量名不为 square 时候

就执行 else 代码块,而 else 里面是只有调用父类方法来赋值这个代码,所以运行到这里就不会调用自身__setattr__ 方法了

而 self.width = value 只会调用自身__setattr__一次,因为 self.width 调用__setattr__的时候,name 是为 width ,不为 square 所以执行的是 else ,而 else 只有 super().__setattr__ 调用父类方法,所以就不会无限递归

而第二个代码,你不管执行 else 还是 if 代码块都只有赋值操作,就会无限调用自身

就好如你赋值调用了 __setattr__ , 结果进入 __setattr__ 方法后还是赋值操作,此时又调用自身,然后又是赋值操作又调用自身.....就永远没有出口了

而 第一个代码 设置了 else 中只有 super().__setattr__ 方法,你可以尝试着在 else 的代码块下 super() 魔法方法前面加个赋值操作,会发现又会是无限递归~


问题2、问题3:


去看源码大部分都是 C 语言的,详见 2 楼

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

使用道具 举报

 楼主| 发表于 2020-7-18 10:49:49 From FishC Mobile | 显示全部楼层
永恒的蓝色梦想 发表于 2020-7-17 23:53
https://github.com/python/cpython
原理就是 魔法(C语言)

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

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-1-19 23:11

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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