多方法和多调度之间有啥区别吗?

Posted

技术标签:

【中文标题】多方法和多调度之间有啥区别吗?【英文标题】:Is there any difference between multimethod and multipledispatch?多方法和多调度之间有什么区别吗? 【发布时间】:2021-06-17 01:31:36 【问题描述】:

我想在 Python 中使用重载。我知道这是不可能的设计(Python 是动态类型语言),关于这个主题有相当好的线程here。这就是为什么我可以使用多重调度(多方法)之类的东西。我正在考虑最好的解决方案,它可以为任何函数或类方法实现多次分派。 是否有任何 Python 原生包含的解决方案?

在我的情况下使用 singledispatch(在 3.4 中添加到 Python)是不够的(因为只考虑第一个参数的类型,我需要 multiple)。

我也知道我可以使用关键字参数(即具有单个函数并通过检查所有参数来驱动行为),但是当你有很多参数和组合时它会变得复杂。

到目前为止,我能够找到这两个库(未包含在 Python 标准库中):

multimethod - 提供了一个装饰器,用于向函数添加多个参数调度,可以使用与 functools.singledispatch 相同的注册样式,与注释一起使用,甚至可以使用带注释的谓词或为类提供特殊的 multimeta 命名空间李> multipledispatch - 还为多参数调度提供了装饰器,不适用于注释,与以前的库相比,我看起来没有那么强大

基本用法如下:

from multimethod import multimethod

@multimethod
def add(x: int, y: int):
    ...

@multimethod
def add(x: str, y: str):
    ...

from multipledispatch import dispatch

@dispatch(int, int)
def add(x, y):
    ...

@dispatch(str, str)
def add(x, y):
    ...

这个功能看起来很相似。

我主要有两个问题:

这些多个调度实现中的一个是否比另一个带来任何(不利)优势?(例如,支持的类型、速度、性能、限制......)它似乎具有可比性,但我不确定这一点,因为后者 (multipledispatch) 似乎更受欢迎(查看 GitHub 星星)但维护较少(我什至会说不太成熟)。 是否有任何其他解决方案(即使包含在 Python 标准库中)?

【问题讨论】:

【参考方案1】:

我为什么多方法比多调度好得多的原因

    我们添加到方法签名中的类型信息,而不是装饰器签名中。这样做更直观、更简洁。 多方法的类型提示与static type support/mypy 的预期相同。现在,类型注释在开发人员中变得很普遍。所以,这对双方都有帮助!多方法类型提示与 mypy 类型提示正式相同 请注意,无论是多方法还是多调度,似乎它们只支持非关键字函数调用。如果您在调用时使用关键字调用函数,则两者都会失败。这令人失望,但他们可能有更多理由这样做。请阅读各个模块的指南。他们谈论基于关键字的函数调用,但不是很清楚,IMO。因此,有时我觉得,在我们的函数中进行简单的类型检查会立即解决所有问题,而不是使用多方法或多分派。毕竟,我想要一流的解决方案,因为函数是一流的对象。 据我所知,语言中没有针对多态性/函数重载的解决方案。但是多方法看起来像 Pythonic 和原生的。我只是喜欢它。此外,multimethod 还支持单次调度,虽然我从来没有觉得需要它。 感谢您的提问。你已经回答了。在阅读您的问题之前,我不知道确切的区别。你已经提供了答案。感谢您的提问。

【讨论】:

至于 3,我能想到的最大的多重分派语言,Julia,也只分派位置参数。原因是模棱两可。采用f(x: int, y: str): return 1f(y: str, x: int): return 2 方法。您可以通过参数顺序 f(1, "one")f("one", 1) 来区分它们,但关键字放弃顺序,因此是模棱两可的 f(y = "one", x = 1)。 “修复”此问题的唯一方法是强制用户遵守某些参数名称顺序。如果我错了,请纠正我,但我也想不出一种具有函数重载的静态语言来处理命名参数。 谢谢你的观点,我绝对同意类型注释的东西。我暂时将您的答案标记为已接受,因为似乎确实没有其他解决方案,并且与多调度库相比,多方法更好。

以上是关于多方法和多调度之间有啥区别吗?的主要内容,如果未能解决你的问题,请参考以下文章

多线程和多进程模式有啥区别

单因素统计和多因素回归分析有啥区别

python中多进程和多线程的区别

麻烦解释一下linux下进程和线程有啥区别和联系,linux下多线程和多进程通信的实现方法,请通俗解释

多进程和多线程有啥区别?

什么是线程、什么又是多线程 它们之间有什么区别呢?