是否有更好的方法来使用 Python 的类型模块为复合类型创建类型别名?

Max*_*ers 4 python typing typechecking

我有一个带一个参数的函数,它应该以 anint或 aNone作为参数。有几种方法可以为这种复合类型创建类型别名:

# test.py
import typing

IntOrNone_1 = typing.TypeVar('IntOrNone_1', int, None)
IntOrNone_2 = typing.Union[int, None]


def my_func1(xyz: IntOrNone_1):
    return xyz

def my_func2(xyz: IntOrNone_2):
    return xyz


my_func1(12)
my_func1(None)
my_func1(13.7)
my_func1('str')

my_func2(12)
my_func2(None)
my_func2(13.7)
my_func2('str')
Run Code Online (Sandbox Code Playgroud)

两种方法都按照我的预期执行,但是,对应的mypy错误略有不同,但基本上具有相同的含义。

test.py:14: 错误:“my_func1”的类型变量“IntOrNone_1”的值不能是“float”

test.py:15: 错误:“my_func1”的类型变量“IntOrNone_1”的值不能是“str”

test.py:19: 错误:“my_func2”的参数 1 具有不兼容的类型“float”;预期“可选[int]”

test.py:20: 错误:“my_func2”的参数 1 具有不兼容的类型“str”;预期“可选[int]”

我倾向于使用第二种方法,因为它还会报告导致错误的参数。

我认为这两种方法确实等效,还是首选其中之一?

Mic*_*x2a 8

这两种方法远非等价。您应该避免将 TypeVar 视为仅仅是别名——相反,它们是更特殊的形式,当您想让您的函数成为generic 时使用

用一个例子来解释什么是“通用函数”是最容易的。假设您要编写一个函数,该函数接受某个对象(任何对象!)并返回另一个完全相同类型的对象。你会怎么做?

我们可以做的一种方法是尝试使用object

def identity(x: object) -> object:
    return x
Run Code Online (Sandbox Code Playgroud)

这让我们很接近,因为我们的identity函数至少可以接受任何字面意思(因为所有类型都继承自objectPython)。然而,这个解决方案是有缺陷的:如果我们传入一个 int ,我们就会返回 out object,这不是我们想要的。

相反,我们需要的是一种让类型检查器理解这两种类型之间存在“关系”的方法。这正是TypeVar派上用场的地方:

T = TypeVar('T')

def identity(x: T) -> T:
    return x
Run Code Online (Sandbox Code Playgroud)

在这种情况下,我们的 TypeVar 'T' 充当“占位符”,可以绑定到我们想要的任何类型。因此,如果我们这样做identity(3)T绑定int- 因此类型检查器将理解返回类型也必须是int

如果我们T在参数类型提示中多次使用,类型检查器将确保每次的类型都相同。


那么,下面的表达式是做什么的?

IntOrNone_1 = typing.TypeVar('IntOrNone_1', int, None)
Run Code Online (Sandbox Code Playgroud)

好吧,事实证明,向我们的特殊占位符类型添加约束有时会很有用。例如,您已经进行了限制IntOrNone_1,使其只能绑定到intNone,而不能绑定到其他类型。


最后,回答你的最后一个问题:在你给出的例子中,你绝对应该使用 Union,而不是 TypeVars。

是否使用联合或联合的类型别名实际上是个人品味的问题,但如果您不需要这种“占位符”或“通用”行为,则不应使用 TypeVars。

如果您想了解有关如何使用 TypeVars 的更多信息,mypy 文档中有一个关于泛型部分。它涵盖了我跳过的几件事,包括如何创建泛型类(不仅仅是泛型函数)。