python 类型提示应该如何要求一个值具有给定的属性?

Rob*_*ton 6 python type-hinting

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

def foo(a: Any):
    return a.bar + a.baz
Run Code Online (Sandbox Code Playgroud)

我想将类型提示从提供和属性的类型提示更改为Any需要(好吧,建议,因为它是类型提示)。应该改成什么?abarbaz

Mic*_*x2a 9

这正是协议的用途。简而言之,Protocols 允许您使用结构化而不是名义子类型化。对于名义子类型,如果 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"))
Run Code Online (Sandbox Code Playgroud)

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

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

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