|
马上注册,结交更多好友,享用更多功能^_^
您需要 登录 才可以下载或查看,没有账号?立即注册
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)
输入: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中查看
问题有些多,请大佬们多些耐心
本帖最后由 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() 魔法方法前面加个赋值操作,会发现又会是无限递归~
去看源码大部分都是 C 语言的,详见 2 楼
|
|