你用对 hasattr 了嘛?

前天帮一个同学DEBUG一个很奇怪的问题,发现了一个Python 2的hasattr一个不适用场景,和大家分享一下。
在日常开发中,判断对象有没有一个属性,你可能这么写:

如果self有redirect _url这个属性,那么请求直接重定向到redirect
_url这个URL上。看起来没有问题对吧,我们写的更完整一些:
看到了吧hassattr的结果是False,也就是说Python 2认为vs这个实例就没有redirect_url属性。但是看代码是有这个property的,只是在计算时由于某些原因抛错误了。
我debug了好久才发现这个原来是这个问题, ‍
赶紧网上搜了下,原来 attrs的作者,Python核心开发在16年就写过一篇文章介绍这个事情:
https://hynek.me/articles/hasattr/

也就是说在Python2,如果你想要判断一个类里面是否有某个property,需要确认它会不会有机会抛错

如果它有可能会异常,那么应该用getattr(让异常发生):
另外在CPython实现,hasattr其实内部还是用了getattr的:

注意 result=PyObject_GetAttr(v,name)
这步,所以getattr要比hasattr更快更直接。
而在Python 3没有这个问题,他会捕获AttributeError返回False,而其他错误直接抛出来:
也就抛出了PyExc_AttributeError会被清除(其他的错误照常raise)

延伸阅读

  1. https://hynek.me/articles/hasattr/
  2. https://github.com/python/cpython/blob/3.7/Objects/object.c#L1231-L1242