告别复杂判断!Python中实现函数重载的终极技巧

引言

说到函数重载,学过 Java 的同学应该不陌生,最常用的地方应该就是打印 log 了,对于不同的参数,调用的是不同的重载函数。那么 Python 如何实现函数重载呢?

重载概念

函数重载是指在同一作用域内,允许多个同名函数存在,但它们的参数列表不同。虽然许多编程语言(如 Java 和 C++)支持函数重载,但 Python 的设计哲学使其不能直接支持这一特性。

不使用重载

先看一个例子,在不使用重载的情况下,实现一个 log 函数:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
from attr import dataclass


@dataclass
class MyException(Exception):
msg: str


def log(message):
if isinstance(message, MyException):
print(message.msg)
elif isinstance(message, str):
print(message)
else:
print(f'invalid message:{message}')


log(MyException('my exception'))
log('str exception')
log(1111)

# 运行结果为:
# my exception
# str exception
# invalid message:1111

不使用重载的话,就要写很多判断的代码,来判断入参的类型。

使用重载

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
from functools import singledispatch

from attr import dataclass


@dataclass
class MyException(Exception):
msg: str


@singledispatch
def log(message):
print(f'invalid message:{message}')


@log.register
def _(message: MyException):
print(message.msg)


@log.register
def _(message: str):
print(message)


log(MyException('my exception'))
log('str exception')
log(1111)

# 运行结果为:
# my exception
# str exception
# invalid message:1111

通过以上代码可以看到,使用重载的情况下,代码简洁了很多,将之前的 if-else 判断都去掉了,每个重载函数根据对应的类型直接输出对应的日志,说明在调用函数时,会自动判断函数的参数类型,然后调用对应的重载函数来执行对应的逻辑。

如果有新增类型的需求,只需要在原有的基础上增加一个重载函数即可,大大简化了新增类型的难度。例如:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
from functools import singledispatch

from attr import dataclass


@dataclass
class MyException(Exception):
msg: str


@singledispatch
def log(message):
print(f'invalid message:{message}')


@log.register
def _(message: MyException):
print(message.msg)


@log.register
def _(message: str):
print(message)


@log.register
def _(message: int):
print(f'int message:{message}')


log(MyException('my exception'))
log('str exception')
log(1111)

# 运行结果为:
# my exception
# str exception
# int message:1111

重载写法

通过以上的代码,总结 Python 通过 functools.singledispatch 进行重载的写法。

  • 首先定义一个函数,加上 @singledispatch 装饰器
  • 然后添加几个以下划线为函数名的函数,因为重载函数的名字都一样,没有必要起其他的名字,用下划线代替即可。
  • 为下划线函数设置对应的函数参数类型,编写函数内容。
  • 调用函数来测试是否生效。

数据类

可能会有小伙伴问,自定义异常上有一个 @dataclass 装饰器,这个是干嘛用的,为什么没有写 __init__() 函数。

这个装饰器是Python3.6 中新引入的一个概念,熟悉 Java 的小伙伴可能会知道,它有点类似于 Java 中的 lombok 中的 @data 注解。它的作用是自动为用户自定义的类添加生成的特殊方法,例如 __init__()__repr__()

1
2
3
4
5
6
7
8
9
10
11
from dataclasses import dataclass

@dataclass
class InventoryItem:
"""Class for keeping track of an item in inventory."""
name: str
unit_price: float
quantity_on_hand: int = 0

def total_cost(self) -> float:
return self.unit_price * self.quantity_on_hand

上面的类不需要单独再写下面的函数,@dataclass 装饰器会自动生成。

1
2
3
4
def __init__(self, name: str, unit_price: float, quantity_on_hand: int = 0):
self.name = name
self.unit_price = unit_price
self.quantity_on_hand = quantity_on_hand

所以 @dataclass 可以帮助我们简化代码,提升开发效率,具体的用法可以参考官方文档

总结

在 Python 中,虽然不支持传统的函数重载,但我们可以通过 functools.singledispatch 方法来实现类似的功能。同时,需要注意一下,只有第一个参数的不同类型会被重载,后面参数的类型变化会被忽略。

本文章首发于个人博客 LLLibra146’s blog
本文作者:LLLibra146
更多文章请关注:qrcode
版权声明:本博客所有文章除特别声明外,均采用 © BY-NC-ND 许可协议。非商用转载请注明出处!严禁商业转载!
本文链接
https://blog.d77.xyz/archives/b708c662.html