python getattribute方法_Python - __getattr__() 和 __getattribute__() 方法的区别

python 再访问属性的方法上定义了__getattr__() 和 __getattribute__() 2种方法,其区别非常细微,但非常重要。

如果某个类定义了 __getattribute__() 方法,在 每次引用属性或方法名称时 Python 都调用它(特殊方法名称除外,因为那样将会导致讨厌的无限循环)。

如果某个类定义了 __getattr__() 方法,Python 将只在正常的位置查询属性时才会调用它。如果实例 x 定义了属性 color, x.color 将 不会 调用x.__getattr__('color');而只会返回 x.color 已定义好的值。

让我们用两个例子来解释一下:

class Dynamo(object):

def __getattr__(self, key):if key == 'color': ① return 'PapayaWhip'

else:raise AttributeError ② >>> dyn = Dynamo() >>> dyn.color ③ 'PapayaWhip' >>> dyn.color = 'LemonChiffon' >>> dyn.color ④ 'LemonChiffon'

属性名称以字符串的形式传入 __getattr()__ 方法。如果名称为 'color',该方法返回一个值。(在此情况下,它只是一个硬编码的字符串,但可以正常地进行某些计算并返回结果。)

如果属性名称未知, __getattr()__ 方法必须引发一个 AttributeError 例外,否则在访问未定义属性时,代码将只会默默地失败。(从技术角度而言,如果方法不引发例外或显式地返回一个值,它将返回 None ——Python 的空值。这意味着 所有 未显式定义的属性将为 None,几乎可以肯定这不是你想看到的。)

dyn 实例没有名为 color 的属性,因此在提供计算值时将调用 __getattr__() 。

在显式地设置 dyn.color 之后,将不再为提供 dyn.color 的值而调用 __getattr__() 方法,因为 dyn.color 已在该实例中定义。

另一方面,__getattribute__() 方法是绝对的、无条件的。

class SuperDynamo(object):

def __getattribute__(self, key):

if key == 'color':

return 'PapayaWhip'

else:

raise AttributeError >>> dyn = SuperDynamo() >>> dyn.color ① 'PapayaWhip' >>> dyn.color = 'LemonChiffon' >>> dyn.color ② 'PapayaWhip'

在获取 dyn.color 的值时将调用 __getattribute__() 方法。

即便已经显式地设置 dyn.color,在获取 dyn.color 的值时, 仍将调用 __getattribute__() 方法。如果存在 __getattribute__() 方法,将在每次查找属性和方法时 无条件地调用 它,哪怕在创建实例之后已经显式地设置了属性。

☞如果定义了类的 __getattribute__() 方法,你可能还想定义一个 __setattr__() 方法,并在两者之间进行协同,以跟踪属性的值。否则,在创建实例之后所设置的值将会消失在黑洞中。

必须特别小心 __getattribute__() 方法,因为 Python 在查找类的方法名称时也将对其进行调用。

class Rastan(object):

def __getattribute__(self, key):raise AttributeError ① def swim(self):

pass >>> hero = Rastan() >>> hero.swim() ② Traceback (most recent call last):

File "", line 1, in

File "", line 3, in __getattribute__

AttributeError

该类定义了一个总是引发 AttributeError 例外的 __getattribute__() 方法。没有属性或方法的查询会成功。

调用 hero.swim() 时,Python 将在 Rastan 类中查找 swim() 方法。该查找将执行整个 __getattribute__()方法,因为所有的属性和方法查找都通过__getattribute__() 方法。在此例中, __getattribute__() 方法引发 AttributeError 例外,因此该方法查找过程将会失败,而方法调用也将失败。

"__getattr__"将会在寻找不到合适的函数或者属性时作为默认被调用到。

通常来说,如果脚本即包含"__getattribute__" ,又包含"__getattr__"的时候,是不会调用到"__getattr__"的,但从上面的例子可以看出,如果"__getattribute__"报错,同样会自动调用到"__getattr__"。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值