与 Python 的流畅界面

sam*_*ba2 3 python dsl fluent-interface

我有一个 Python 函数“send_message”,它接受三个参数:

send_message("i like windmills", to="INBOX", from="OUTBOX")
Run Code Online (Sandbox Code Playgroud)

我正在考虑在它上面放置一个流畅的界面。理想情况下,我想编写以下任何内容:

send_message("i like windmills").to("INBOX").from("OUTBOX")

send_message("i like windmills").from("OUTBOX").to("INBOX")

# The `to()` information is mandatory but the `from()` is not (as with real letters), so this one would also be a valid call:
send_message("i like windmills").to("INBOX")
Run Code Online (Sandbox Code Playgroud)

任何想法如何实现这一点或类似的东西?

我理解让对象的方法返回“self”的一般方法,但在我的理解中,这将导致如下结果:

message = Message("i like windmills")
message.to("INBOX").from("OUTBOX").send()
Run Code Online (Sandbox Code Playgroud)

但是这个不如前面的例子好,那么我实际上更喜欢带有命名参数的原始版本。

任何帮助表示赞赏。

The*_*ter 6

可以通过这种方式完成,我不确定是否有更好的方法,因为这是我的第一次尝试。祝你好运!

DEFAULT_SENDER = 'my_address'
#Because the sender object is optional I assume you have a default sender

class Send_message(object):
    def __init__(self, message):
        self.message = message
        self.sender = None
        self.receiver = None
        self.method = None

    def to(self, receiver):
        self.receiver = receiver
        self.method = self.send()
        return self

    def _from(self, sender):
        self.sender = sender
        self.method = self.send()
        return self

    def __call__(self):
        if self.method:
            return self.method()
        return None

    def send(self):
        if self.receiver:
            if not self.sender:
                self.sender = DEFAULT_SENDER

            return lambda:actual_message_code(self.message, self.sender, self.receiver)


def actual_message_code(message, sender, receiver):
    print "Sent '{}' from: {} to {}.".format(message, sender, receiver)



Send_message("Hello")._from('TheLazyScripter').to('samba2')()
Send_message("Hello").to('samba2')._from('TheLazyScripter')()
Send_message("Hello").to('samba2')()

#Only change in actual calling is the trailing ()
Run Code Online (Sandbox Code Playgroud)

通过实现该__call__方法,我们可以知道我们何时处于调用链的末尾。这当然会添加尾随()调用。并要求您更改指向实际消息传递方法和默认发件人变量的指针,但我认为这将是实现目标的最简单方法,而实际上不知道链何时结束。