Python 类型提示应如何要求值具有给定属性?

问题描述 投票:0回答:2

假设我有一个像这样的简单函数:

def foo(a: Any):
    return a.bar + a.baz

我想将类型提示从

Any
更改为需要(好吧,建议,因为它是一种类型 hint
a
提供
bar
baz
属性。 应该改成什么?

python python-typing
2个回答
32
投票

这正是协议的用途。简而言之,协议允许您使用“结构”而不是“名义”子类型。对于名义子类型,如果 A 显式继承或扩展 B,则类型 A 是 B 的子类型。对于结构子类型,如果类型 A 具有与 B 相同的方法和属性“签名”(有一些限制),则类型 A 是 B 的子类型。 例如: # If you're using Python 3.8+ from typing import Protocol # If you need to support older versions of Python, # pip-install the 'typing_extensions' module and do: from typing_extensions import Protocol class SupportsBarBaz(Protocol): bar: int baz: int class MyUnrelatedClass1: def __init__(self, bar: int, baz: int) -> None: self.bar = bar self.baz = baz class MyUnrelatedClass2: def __init__(self, bar: int, baz: int, blah: str) -> None: self.bar = bar self.baz = baz self.blah = blah class MyUnrelatedClass3: def __init__(self, bar: str, baz: str, blah: str) -> None: self.bar = bar self.baz = baz self.blah = blah def foo(a: SupportsBarBaz) -> int: return a.bar + a.baz # These both type-check, even though there's no explicit relationship # between 'SupportsBarBaz' and these two classes foo(MyUnrelatedClass1(1, 2)) foo(MyUnrelatedClass2(1, 2, "abc")) # But this doesn't type-check, since 'bar' and 'baz' are both strs here foo(MyUnrelatedClass3("a", "b", "c"))

您可以在

mypy 文档
中找到有关使用协议的更多信息。该页面中的信息全部符合 PEP,因此那里的信息应该全部适用于其他类型检查器,假设他们已经完成了自己对协议的支持。

您还可以在 typeshed(Python 标准库的类型提示存储库)中找到使用协议的稍微复杂的示例。

不过,我认为只有当您真正打算在代码中使用静态分析时,这一切才重要。如果没有,您可以做一些更简单的事情,只需为 Any 定义一个自定义类型别名,记录该别名“应该”的含义,并使用该别名而不是成熟的协议。对于静态分析/自动完成工具等来说,该别名几乎完全无用,但人们通常在阅读评论时没有问题。

类型提示只能引用一个类,因此创建一个抽象类


1
投票

并声明

f(a: MyType)

    


© www.soinside.com 2019 - 2024. All rights reserved.