python hasattr ()

不要使用 python 中的 hasattr() 除非你只在python3环境中使用它并且知道它的内部原理

总的来说,不要以如下方式使用

if hasattr(x, "y"):
    print x.y
else:
    print "no y!"

我们可以使用如下方式进行代替

try:
    print x.y
except AttributeError:
    print "no y!"

或者:

y = getattr(x, "y", None)
if y is not None:
    print y
else:
    print "no y!"

hasattr() 其实并不比 getattr() 要快, 他们使用完全相同的方式去查找属性,只是hasattr()抛弃了结果。

为什么会出现这种情况:

在python 2 中, hasattr() 使用类似下面代码的方式进行运作

try:
    print x.y
except:
    print "no y!"

这根本就不是你想要的, 他会将属性里面的异常全部吞掉

class C(object):
    @property
    def y(self):
        0/0

    def x_y(self):
        0/0

hasattr(C(), "y")
> False

hasattr(C(), "x_y")
> True

当你的东西使用第三方包的时候,你无法确定你使用的属性是否是一个 property(或者在若干年之后变成一个property), 这是非常危险的

在python 3中,运作是正常的。

class C:
    @property
    def y(self):
        0/0

>>> hasattr(C(), "y")
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
  File "<stdin>", line 4, in y
ZeroDivisionError: division by zero

但是你真的期望 hasattr() 去抛出一个异常吗?

原文:https://hynek.me/articles/hasattr/

本文章首发在 PythonCaff