rau*_*ido 11 python cpython python-2.7 python-3.x
我最近一直在阅读一些关于hasattr的推文和python文档,它说:
hasattr(对象,名称)
参数是一个对象和一个字符串.如果字符串是>>对象属性之一的名称,则结果为True,否则返回False.(这是通过调用getattr(object,name)并查看它是否引发AttributeError来实现的.)
Python中有一句座右铭,就是我通常同意的许可更容易要求宽恕.
在这种情况下,我尝试使用一个非常简单的python代码进行性能测试:
import timeit
definition="""\
class A(object):
a = 1
a = A()
"""
stm="""\
hasattr(a, 'a')
"""
print timeit.timeit(stmt=stm, setup=definition, number=10000000)
stm="""\
getattr(a, 'a')
"""
print timeit.timeit(stmt=stm, setup=definition, number=10000000)
Run Code Online (Sandbox Code Playgroud)
结果如下:
$ python test.py
hasattr(a, 'a')
1.26515984535
getattr(a, 'a')
1.32518696785
Run Code Online (Sandbox Code Playgroud)
如果属性不存在且getattr和hasattr之间的差异更大,我也尝试了会发生什么.所以我到目前为止看到的是getattr比hasattr慢,但在文档中它说它调用了getattr.
我搜索了hasattr和getattr的CPython实现,似乎都调用了下一个函数:
v = PyObject_GetAttr(v, name);
Run Code Online (Sandbox Code Playgroud)
但是getattr中的样板比hasattr中的样板更多,这可能使它更慢.
有谁知道为什么在文档中我们说hasattr调用getattr并且我们似乎鼓励用户使用getattr而不是hasattr,当它真的不是由于性能?只是因为它更pythonic?
也许我在测试中做错了什么:)
谢谢,
劳尔
Ant*_*ala 16
文档并不鼓励,文档只是说明了.这样hasattr
实现,并且AttributeError
从属性getter中抛出一个可以使它看起来像属性不存在.这是一个重要的细节,这就是文档中明确说明的原因.考虑例如这段代码:
class Spam(object):
sausages = False
@property
def eggs(self):
if self.sausages:
return 42
raise AttributeError("No eggs without sausages")
@property
def invalid(self):
return self.foobar
spam = Spam()
print(hasattr(Spam, 'eggs'))
print(hasattr(spam, 'eggs'))
spam.sausages = True
print(hasattr(spam, 'eggs'))
print(hasattr(spam, 'invalid'))
Run Code Online (Sandbox Code Playgroud)
结果是
True
False
True
False
Run Code Online (Sandbox Code Playgroud)
这是Spam
类有一个属性描述符eggs
,但由于getter引发AttributeError
if not self.sausages
,那么该类的实例不是" hasattr
" eggs
.
除此之外,hasattr
仅在您不需要该值时使用 ; 如果你需要这个值,使用getattr
2个参数并捕获异常或3个参数,第三个是合理的默认值.
结果使用getattr()
(2.7.9):
>>> spam = Spam()
>>> print(getattr(Spam, 'eggs'))
<property object at 0x01E2A570>
>>> print(getattr(spam, 'eggs'))
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 7, in eggs
AttributeError: No eggs without sausages
>>> spam.sausages = True
>>> print(getattr(spam, 'eggs'))
42
>>> print(getattr(spam, 'invalid'))
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
File "<stdin>", line 10, in invalid
AttributeError: 'Spam' object has no attribute 'invalid'
>>>
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
18516 次 |
最近记录: |