为什么 Python 不支持函数重载?而其他语言大都支持?

关注者
483
被浏览
331,518

33 个回答

这个问题,最近在 cpyug 上面讨论得很火热。我简要概括一下。

为了考虑为什么 python 不提供函数重载,首先我们要研究为什么需要提供函数重载。

函数重载主要是为了解决两个问题。

1。可变参数类型。

2。可变参数个数。

另外,一个基本的设计原则是,仅仅当两个函数除了参数类型和参数个数不同以外,其功能是完全相同的,此时才使用函数重载,如果两个函数的功能其实不同,那么不应当使用重载,而应当使用一个名字不同的函数。

好吧,那么对于情况 1 ,函数功能相同,但是参数类型不同,python 如何处理?答案是根本不需要处理,因为 python 可以接受任何类型的参数,如果函数的功能相同,那么不同的参数类型在 python 中很可能是相同的代码,没有必要做成两个不同函数。

那么对于情况 2 ,函数功能相同,但参数个数不同,python 如何处理?大家知道,答案就是缺省参数。对那些缺少的参数设定为缺省参数即可解决问题。因为你假设函数功能相同,那么那些缺少的参数终归是需要用的。

好了,鉴于情况 1 跟 情况 2 都有了解决方案,python 自然就不需要函数重载了。

我补充一点,Python3是可以通过metaclass + parameter annotation使某个类假装支持function overloading

class Spam(metaclass=MultipleMeta):
    def bar(self, x:int, y:int):
        print('Bar 1:', x, y)
    def bar(self, s:str, n:int = 0):
        print('Bar 2:', s, n)

# Example: overloaded __init__
import time
class Date(metaclass=MultipleMeta):
    def __init__(self, year:int, month:int, day:int):
        self.year = year
        self.month = month
        self.day = day

    def __init__(self):
        t = time.localtime()
        self.__init__(t.tm_year, t.tm_mon, t.tm_mday)

其中MultipleMeta的实现可以参考

chimera.labs.oreilly.com

,简单来说就是实现一个attribute与functions的一对多关系。

@pansz

对于function overloading的总结已经很好了。就我个人理解,functino overloading无非就是想要解决取名难的问题(代码耦合的问题可以通过重构解决),但事实上这也不算是什么难题,而且Python的parameter设定也足以应付绝大多数需求了,所以社区基于效率与语言复杂度的考虑,干脆就不支持function overloading了。当然以上都是我瞎猜的。