Python中的“method_missing”与Ruby的相似性

17

请问您能否描述一下您所寻找的确切机制? - Zaur Nasibov
我正在尝试在Python中实现类似于Rails动态查找器的功能。 - Zeck
@Zeck,我的意思是,你能否解释一下,对于非Ruby人员,你正在寻找什么? - Zaur Nasibov
2个回答

48
正如其他人所提到的,在Python中,当你执行o.f(x)时,它实际上是一个两步操作:首先,获取of属性,然后用参数x调用它。失败的是第一步,因为没有f属性,而正是这一步调用了Python的魔术方法__getattr__
因此,您必须实现__getattr__,并且它返回的内容必须是可调用的。请记住,如果您尝试获取o.some_data_that_doesnt_exist,同样会调用相同的__getattr__,它将无法知道正在寻找的是“数据”属性还是“方法”。
下面是一个返回可调用函数的示例:
class MyRubylikeThing(object):
    #...

    def __getattr__(self, name):
        def _missing(*args, **kwargs):
            print "A missing method was called."
            print "The object was %r, the method was %r. " % (self, name)
            print "It was called with %r and %r as arguments" % (args, kwargs)
        return _missing

r = MyRubylikeThing()
r.hello("there", "world", also="bye")

生成:

A missing method was called.
The object was <__main__.MyRubylikeThing object at 0x01FA5940>, the method was 'hello'.
It was called with ('there', 'world') and {'also': 'bye'} as arguments

这只能覆盖缺失的方法。如果我想要覆盖缺失的成员和方法怎么办?在 "def getattr(self, name)" 中,name 属性只是一个字符串,无法告诉我这个名称对应的属性是方法还是成员。 - songzhw

1
你可以重载__getattr__方法并从中返回一个可调用对象。请注意,在属性查找期间,你无法决定所请求的属性是否打算被调用,因为Python会分两步完成此操作。

谢谢大家。我在这里找到了一个解决方案(https://dev59.com/W0fRa4cB1Zd3GeqP_r6V#913438)。但我想知道如何传递参数? - Zeck

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接