详解Python 函数如何重载?
什么是函数重载?简单的理解,支持多个同名函数的定义,只是参数的个数或者类型不同,在调用的时候,解释器会根据参数的个数或者类型,调用相应的函数。
重载这个特性在很多语言中都有实现,比如C++、Java等,而Python并不支持。这篇文章呢,通过一些小技巧,可以让Python支持类似的功能。
参数个数不同的情形
先看看这种情况下C++是怎么实现重载的
#includeusingnamespacestd; intfunc(inta) { cout<<'Oneparameter'< 如果Python按类似的方式定义函数的话,不会报错,只是后面的函数定义会覆盖前面的,达不到重载的效果。
>>>deffunc(a): ...print('Oneparameter') ... >>>deffunc(a,b): ...print('Twoparameters') ... >>>deffunc(a,b,c): ...print('Threeparameters') ... >>>func(1) Traceback(mostrecentcalllast): File"",line1,in TypeError:func()missing2requiredpositionalarguments:'b'and'c' >>>func(1,2) Traceback(mostrecentcalllast): File" ",line1,in TypeError:func()missing1requiredpositionalargument:'c' >>>func(1,2,3) Threeparameters 但是我们知道,Python函数的形参十分灵活,我们可以只定义一个函数来实现相同的功能,就像这样
>>>deffunc(*args): ...iflen(args)==1: ...print('Oneparameter') ...eliflen(args)==2: ...print('Twoparameters') ...eliflen(args)==3: ...print('Threeparameters') ...else: ...print('Error') ... >>>func(1) Oneparameter >>>func(1,2) Twoparameters >>>func(1,2,3) Threeparameters >>>func(1,2,3,4) Error参数类型不同的情形
同样,先看下当前情况下C++的重载是怎么实现的
#includeusingnamespacestd; intfunc(inta) { cout<<'Int:'< 代码中,func支持两种类型的参数:整形和浮点型。调用时,解释器会根据参数类型去寻找合适的函数。Python要实现类似的功能,需要借助functools.singledispatch装饰器。
fromfunctoolsimportsingledispatch @singledispatch deffunc(a): print(f'Other:{a}') @func.register(int) def_(a): print(f'Int:{a}') @func.register(float) def_(a): print(f'Float:{a}') if__name__=='__main__': func('zzz') func(1) func(1.2)func函数被functools.singledispatch装饰后,又根据不同的参数类型绑定了另外两个函数。当参数类型为整形或者浮点型时,调用绑定的对应的某个函数,否则,调用自身。
执行结果
Other:zzz
Int:1
Float:1.2需要注意的是,这种方式只能够根据第一个参数的类型去确定最后调用的函数。
关于singledispatch的更多细节请看官方文档
https://docs.python.org/3.6/library/functools.html#functools.singledispatch
注意:函数返回值不同也是重载的一种情况,暂时没有比较好的Python实现方式,所以没有提及
个人觉得,重载就是为了语言的灵活性而设计的,而Python函数本来就有不少巧妙的设计,这个时候去仿这个技术,其实没有多大必要,而且感觉有些违背Python的哲学。所以,本文更多的是在讲如何模仿,而对于重载的使用场景并没有作多少说明。
以上所述是小编给大家介绍的Python函数重载详解整合,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对毛票票网站的支持!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。