附着的小菜鸡 发表于 2020-7-17 23:50:38

关于魔法方法__setattr__的疑问

正常编写的代码:
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)
       
输入: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中查看

问题有些多,请大佬们多些耐心{:5_109:}


永恒的蓝色梦想 发表于 2020-7-17 23:53:24

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

问题3:我应该如何才能看到魔法方法的实现代码?最好是在pycharm中查看https://github.com/python/cpython
原理就是 魔法(C语言)

永恒的蓝色梦想 发表于 2020-7-17 23:57:48

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

附着的小菜鸡 发表于 2020-7-18 00:04:10

永恒的蓝色梦想 发表于 2020-7-17 23:57
因为 self.name=value 还是需要调用 __setattr__ 自己。

为什么self.width=value 没有调用__setattr__ 自己呢?这里不是也是改变了属性的值吗?

Twilight6 发表于 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 楼

附着的小菜鸡 发表于 2020-7-18 10:49:49

永恒的蓝色梦想 发表于 2020-7-17 23:53
https://github.com/python/cpython
原理就是 魔法(C语言)

感谢
页: [1]
查看完整版本: 关于魔法方法__setattr__的疑问