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]”
我倾向于使用第二种方法,因为它还会报告导致错误的参数。
我认为这两种方法确实等效,还是首选其中之一?
这两种方法远非等价。您应该避免将 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,使其只能绑定到int或None,而不能绑定到其他类型。
最后,回答你的最后一个问题:在你给出的例子中,你绝对应该使用 Union,而不是 TypeVars。
是否使用联合或联合的类型别名实际上是个人品味的问题,但如果您不需要这种“占位符”或“通用”行为,则不应使用 TypeVars。
如果您想了解有关如何使用 TypeVars 的更多信息,mypy 文档中有一个关于泛型的部分。它涵盖了我跳过的几件事,包括如何创建泛型类(不仅仅是泛型函数)。
| 归档时间: |
|
| 查看次数: |
1699 次 |
| 最近记录: |