Python 的动态特性和对鸭子类型的支持长期以来因其灵活性而受到称赞。然而,随着代码库变得越来越大、越来越复杂,静态类型检查的好处变得越来越明显。但是我们如何协调鸭子类型的灵活性和静态类型检查的安全性呢?进入Python的Protocol类。
在本教程中,您将学习:
鸭子类型是一种编程概念,其中对象的类型或类不如它定义的方法重要。它基于这样的想法:“如果它看起来像鸭子,像鸭子一样游泳,像鸭子一样嘎嘎叫,那么它可能就是一只鸭子。”
在 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 参数没有类型提示,这在鸭子类型代码中很常见,但可能会导致较大代码库中出现问题。
鸭子打字有几个优点:
但是,它也有一些缺点:
解决这些问题的一种方法是使用抽象基类(ABC)。这是一个例子:
from abc import ABC, abstractmethod class Quacker(ABC): @abstractmethod def quack(self): pass class 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 3.8引入了Protocol类,它允许我们定义接口而不需要继承。下面是我们如何使用它:
from typing import Protocol class 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)
让我们来分解一下:
这种方法给我们带来了几个好处:
这是一个更复杂的示例,展示了协议如何根据需要(形状)变得复杂,同时保持域类(圆形、矩形)平坦:
from typing import Protocol, List class Drawable(Protocol): def draw(self): ... class Resizable(Protocol): def resize(self, factor: float): ... class Shape(Drawable, Resizable, Protocol): pass def process_shapes(shapes: List[Shape]): for shape in shapes: shape.draw() shape.resize(2.0) # Example usage class 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 inheritance shapes: List[Shape] = [Circle(), Rectangle()] process_shapes(shapes)
在此示例中,Circle 和 Rectangle 不继承自 Shape 或任何其他类。他们只是实现所需的方法(绘制和调整大小)。得益于 Shape 协议,process_shapes 函数可以与任何具有这些方法的对象一起使用。
Python 中的协议提供了一种将静态类型引入鸭子类型代码的强大方法。它们允许我们在类型系统中指定接口,而不需要继承,保持鸭子类型的灵活性,同时添加静态类型检查的好处,
通过使用协议,您可以:
如果您想了解有关 Python 中的协议和类型提示的更多信息,请查看有关打字模块的官方 Python 文档,或探索 mypy 等高级静态类型检查工具。
祝你编码愉快,愿你的鸭子总是因类型安全而嘎嘎叫!
您可以在这里找到更多我的内容,包括我的时事通讯
以上是鸭子类型与类型提示的结合:在 Python 中使用协议的详细内容。更多信息请关注PHP中文网其他相关文章!