8

我有一个使用__getattr__. 属性可能很昂贵,例如执行查询。现在,我正在使用一个库,它在实际获取对象之前检查我的对象是否具有该属性。

结果,一个查询被执行了两次而不是一次。当然,实际执行__getattr__以真正知道属性是否存在是有意义的。

class C(object):
    def __getattr__(self, name):
        print "I was accessed"
        return 'ok'

c = C()
hasattr(c, 'hello')

有什么办法可以防止这种情况发生吗?

如果支持 Python,__hasattr__那么我可以简单地检查查询是否存在,而不是实际运行它。

我可以创建一个缓存,但它很重,因为查询可能有参数。当然,服务器可能会自己缓存查询并将问题最小化,但是如果查询返回大量数据,它仍然很重。

有任何想法吗?

4

2 回答 2

5

虽然最初我不喜欢猴子修补的想法,总的来说是一个“坏主意”,但我在 1999 年遇到了一个非常巧妙的解决方案!

http://code.activestate.com/lists/python-list/14972/

def hasattr(o, a, orig_hasattr=hasattr):
    if orig_hasattr(o, "__hasattr__"):
        return o.__hasattr__(a)
    return orig_hasattr(o, a)

__builtins__.hasattr = hasattr

从本质上讲,它在 Python 中创建了对的支持__hasattr__,这是我最初认为的最佳解决方案。

于 2015-05-20T12:28:56.953 回答
0

我认为这通常可能是一个不好的模式,但你总是可以检查一个对象的底层__dict__.

In [1]: class A(object):
   ....:     @property
   ....:     def wha(self):
   ....:         print "was accessed"
   ....:

In [2]: A.__dict__
Out[2]:
<dictproxy {'__dict__': <attribute '__dict__' of 'A' objects>,
'__doc__': None,
'__module__': '__main__',
'__weakref__': <attribute '__weakref__' of 'A' objects>,
'wha': <property at 0x10f6b11b0>}>

In [3]: a = A()

In [4]: "wha" in a.__class__.__dict__
Out[4]: True
于 2015-05-17T18:22:38.773 回答