鱼C论坛

 找回密码
 立即注册
查看: 925|回复: 7

[已解决]关于__setattr__

[复制链接]
发表于 2020-6-22 17:53:37 From FishC Mobile | 显示全部楼层 |阅读模式

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

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

x
class Rectangle:
    def __init__(self,width=0,height=0):
        self.width=width
        self.height=height
    def __setattr__(self,name,value):
        if name=='square':
            self.width=value
            self.height=value
        else:
            super().__setattr__(name,value)
    def getAre(self):
        return self.width*self.height

r1=Rectangle('square',3)
print(r1.getAre())
为什么代码跑起来结果是打印了三次square
最佳答案
2020-6-22 19:16:31



为什么打印三次呢,因为返回的是 self.width*self.height ,而你的传入参数是 'square'  和 3 ,也就是:
self.width = 'square'
self.height = 3

所以字符串 * 3 就是将三个相同的字符串拼接一起,所以返回了3次的 square


第二个问题:

你可能没理解 __setattr__(self,name,value) 里面的  name 参数 和 value 参数代表的是什么

name 代表的就是赋值的变量名, value 代表的是 赋值的值,简单点说就是 等号左边的是name 右边的是 value

然后当你 r1=Rectangle('square',3)  将实例化对象 r1  会自动调用 __init__ 方法

而 __init__ 方法里面有两个赋值操作:
self.width=width
self.height=height

这两个赋值操作会调用直接的 __setattr__ 先来看第一个 self.width=width  这里调用了自己的 __setattr__ 方法

name 指的就是 等号左边的 width ,而 value 是等号右边的 就是你传入的 'square' 字符串,进入 __setattr__ 方法 进行判断 name 是否等于 'square'

因为这里的 name = 'width' 不等于 'square' 所以不满足 if 条件执行 else

else 用super 调用基类的 __setattr__方法 来给 width 赋值为 'square'

同理 self.height=height 也是一样的

所以不会导致 无限递归调用 __setattr__方法




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

使用道具 举报

 楼主| 发表于 2020-6-22 17:56:16 From FishC Mobile | 显示全部楼层
还有为什么else上面的赋值操作没有改,程序却没有无限递归,
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-6-22 17:59:07 | 显示全部楼层
__init__是实例化时触发的,__setattr__是给实例属性赋值时触发的...
所以,self.width是"square",self.height是3
也就是把square重复三遍。
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-6-22 18:09:41 | 显示全部楼层
楼上正解
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-6-22 18:24:38 From FishC Mobile | 显示全部楼层
qiuyouzhi 发表于 2020-6-22 17:59
__init__是实例化时触发的,__setattr__是给实例属性赋值时触发的...
所以,self.width是"square",self.h ...

意思是实例赋值时,__setattr__的参数会自动传入并覆盖__init__的参数?
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-6-22 18:27:24 From FishC Mobile | 显示全部楼层
qiuyouzhi 发表于 2020-6-22 17:59
__init__是实例化时触发的,__setattr__是给实例属性赋值时触发的...
所以,self.width是"square",self.h ...

__setattr__下面有赋值啊,又为什么没用
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-6-22 18:49:00 | 显示全部楼层
tiger吴 发表于 2020-6-22 18:27
__setattr__下面有赋值啊,又为什么没用

__setattr__根本就没有被触发...
想知道小甲鱼最近在做啥?请访问 -> ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-6-22 19:16:31 | 显示全部楼层    本楼为最佳答案   



为什么打印三次呢,因为返回的是 self.width*self.height ,而你的传入参数是 'square'  和 3 ,也就是:
self.width = 'square'
self.height = 3

所以字符串 * 3 就是将三个相同的字符串拼接一起,所以返回了3次的 square


第二个问题:

你可能没理解 __setattr__(self,name,value) 里面的  name 参数 和 value 参数代表的是什么

name 代表的就是赋值的变量名, value 代表的是 赋值的值,简单点说就是 等号左边的是name 右边的是 value

然后当你 r1=Rectangle('square',3)  将实例化对象 r1  会自动调用 __init__ 方法

而 __init__ 方法里面有两个赋值操作:
self.width=width
self.height=height

这两个赋值操作会调用直接的 __setattr__ 先来看第一个 self.width=width  这里调用了自己的 __setattr__ 方法

name 指的就是 等号左边的 width ,而 value 是等号右边的 就是你传入的 'square' 字符串,进入 __setattr__ 方法 进行判断 name 是否等于 'square'

因为这里的 name = 'width' 不等于 'square' 所以不满足 if 条件执行 else

else 用super 调用基类的 __setattr__方法 来给 width 赋值为 'square'

同理 self.height=height 也是一样的

所以不会导致 无限递归调用 __setattr__方法




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

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-1-20 12:10

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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