鸭子类型遇到类型提示:在 Python 中使用协议
python 的动态特性和对鸭子类型的支持长期以来因其灵活性而受到称赞。然而,随着代码库变得越来越大、越来越复杂,静态类型检查的好处变得越来越明显。但是我们如何协调鸭子类型的灵活性和静态类型检查的安全性呢?进入python的protocol类。
在本教程中,您将学习:
- 什么是鸭子类型以及 python 中如何支持它
- 鸭子打字的优点和缺点
- 抽象基类(abc)如何尝试解决打字问题
- 如何使用协议来获得两全其美的效果:通过静态类型检查实现鸭子类型灵活性
了解鸭子类型
鸭子类型是一种编程概念,其中对象的类型或类不如它定义的方法重要。它基于这样的想法:“如果它看起来像鸭子,像鸭子一样游泳,像鸭子一样嘎嘎叫,那么它可能就是一只鸭子。”
在 python 中,完全支持鸭子类型。例如:
class duck: def quack(self): print("quack!")class person: def quack(self): print("i'm imitating a duck!")def make_it_quack(thing): # note: no type hint here thing.quack()duck = duck()person = person()make_it_quack(duck) # output: quack!make_it_quack(person) # output: i'm imitating a duck!
在这个例子中,make_it_quack 不关心事物的类型。它只关心这个东西有一个江湖方法。请注意,thing 参数没有类型提示,这在鸭子类型代码中很常见,但可能会导致较大代码库中出现问题。
立即学习“Python免费学习笔记(深入)”;
鸭子打字的优点和缺点
鸭子打字有几个优点:
- 灵活性:它允许更灵活的代码,不依赖于特定类型。
- 更轻松的代码重用:您可以在新上下文中使用现有的类而无需修改。
- 强调行为:它关注对象可以做什么,而不是它是什么。
但是,它也有一些缺点:
- 缺乏清晰度:可能不清楚对象需要实现哪些方法。
- 运行时错误:与类型相关的错误仅在运行时捕获。
- 较少的 ide 支持:ide 很难提供准确的自动完成和错误检查。
abc 解决方案
解决这些问题的一种方法是使用抽象基类(abc)。这是一个例子:
from abc import abc, abstractmethodclass quacker(abc): @abstractmethod def quack(self): passclass duck(quacker): def quack(self): print("quack!")class person(quacker): def quack(self): print("i'm imitating a duck!")def make_it_quack(thing: quacker): thing.quack()duck = duck()person = person()make_it_quack(duck)make_it_quack(person)
虽然这种方法提供了更好的类型检查和更清晰的接口,但它也有缺点:
- 它需要继承,这可能会导致不灵活的层次结构。
- 它不适用于您无法修改的现有类。
- 这违背了python的“鸭子打字”哲学。
协议:两全其美
python 3.8引入了protocol类,它允许我们定义接口而不需要继承。以下是我们如何使用它:
from typing import protocolclass quacker(protocol): def quack(self):...class duck: def quack(self): print("quack!")class person: def quack(self): print("i'm imitating a duck!")def make_it_quack(thing: quacker): thing.quack()duck = duck()person = person()make_it_quack(duck)make_it_quack(person)
让我们来分解一下:
- 我们定义了一个quacker协议,指定了我们期望的接口。
- 我们的 duck 和 person 类不需要继承任何东西。
- 我们可以使用 make_it_quack 的类型提示来指定它需要 quacker。
这种方法给我们带来了几个好处:
- 静态类型检查:ide 和类型检查器可以在运行前捕获错误。
- 不需要继承:现有的类只要有正确的方法就可以工作。
- 清晰的接口:协议明确定义了期望的方法。
这是一个更复杂的示例,展示了协议如何根据需要(形状)变得复杂,同时保持域类(圆形、矩形)平坦:
from typing import Protocol, Listclass Drawable(Protocol): def draw(self): ...class Resizable(Protocol): def resize(self, factor: float): ...class Shape(Drawable, Resizable, Protocol): passdef process_shapes(shapes: List[Shape]): for shape in shapes: shape.draw() shape.resize(2.0)# Example usageclass Circle: def draw(self): print("Drawing a circle") def resize(self, factor: float): print(f"Resizing circle by factor {factor}")class Rectangle: def draw(self): print("Drawing a rectangle") def resize(self, factor: float): print(f"Resizing rectangle by factor {factor}")# This works with any class that has draw and resize methods,# regardless of its actual type or inheritanceshapes: List[Shape] = [Circle(), Rectangle()]process_shapes(shapes)
在此示例中,circle 和 rectangle 不继承自 shape 或任何其他类。他们只是实现所需的方法(绘制和调整大小)。得益于 shape 协议,process_shapes 函数可以与任何具有这些方法的对象一起使用。
概括
python 中的协议提供了一种将静态类型引入鸭子类型代码的强大方法。它们允许我们在类型系统中指定接口,而不需要继承,保持鸭子类型的灵活性,同时增加静态类型检查的好处,
通过使用协议,您可以:
- 为您的代码定义清晰的接口
- 获得更好的 ide,(静态类型检查),支持并更早捕获错误
- 保持鸭子打字的灵活性
- 利用类型检查来检查您无法修改的类。
如果您想了解有关 python 中的协议和类型提示的更多信息,请查看有关类型模块的 python 官方文档,或探索 mypy 等高级静态类型检查工具。
快乐编码,愿你的鸭子总是因类型安全而嘎嘎叫!
您可以在这里找到更多我的内容,包括我的时事通讯