0

0

鸭子类型遇到类型提示:在 Python 中使用协议

PHPz

PHPz

发布时间:2024-07-31 16:22:58

|

779人浏览过

|

来源于dev.to

转载

鸭子类型遇到类型提示:在 python 中使用协议

python 的动态特性和对鸭子类型的支持长期以来因其灵活性而受到称赞。然而,随着代码库变得越来越大、越来越复杂,静态类型检查的好处变得越来越明显。但是我们如何协调鸭子类型的灵活性和静态类型检查的安全性呢?进入python的protocol类。

在本教程中,您将学习:

  1. 什么是鸭子类型以及 python 中如何支持它
  2. 鸭子打字的优点和缺点
  3. 抽象基类(abc)如何尝试解决打字问题
  4. 如何使用协议来获得两全其美的效果:通过静态类型检查实现鸭子类型灵活性

了解鸭子类型

鸭子类型是一种编程概念,其中对象的类型或类不如它定义的方法重要。它基于这样的想法:“如果它看起来像鸭子,像鸭子一样游泳,像鸭子一样嘎嘎叫,那么它可能就是一只鸭子。”

在 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免费学习笔记(深入)”;

鸭子打字的优点和缺点

鸭子打字有几个优点:

  1. 灵活性:它允许更灵活的代码,不依赖于特定类型。
  2. 更轻松的代码重用:您可以在新上下文中使用现有的类而无需修改。
  3. 强调行为:它关注对象可以做什么,而不是它是什么。

但是,它也有一些缺点:

  1. 缺乏清晰度:可能不清楚对象需要实现哪些方法。
  2. 运行时错误:与类型相关的错误仅在运行时捕获。
  3. 较少的 ide 支持:ide 很难提供准确的自动完成和错误检查。

abc 解决方案

解决这些问题的一种方法是使用抽象基类(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)

虽然这种方法提供了更好的类型检查和更清晰的接口,但它也有缺点:

  1. 它需要继承,这可能会导致不灵活的层次结构。
  2. 它不适用于您无法修改的现有类。
  3. 这违背了python的“鸭子打字”哲学。

协议:两全其美

python 3.8引入了protocol类,它允许我们定义接口而不需要继承。以下是我们如何使用它:

MyMap AI
MyMap AI

使用AI将想法转化为图表

下载
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)

让我们来分解一下:

  1. 我们定义了一个quacker协议,指定了我们期望的接口。
  2. 我们的 duck 和 person 类不需要继承任何东西。
  3. 我们可以使用 make_it_quack 的类型提示来指定它需要 quacker。

这种方法给我们带来了几个好处:

  1. 静态类型检查:ide 和类型检查器可以在运行前捕获错误。
  2. 不需要继承:现有的类只要有正确的方法就可以工作。
  3. 清晰的接口:协议明确定义了期望的方法。

这是一个更复杂的示例,展示了协议如何根据需要(形状)变得复杂,同时保持域类(圆形、矩形)平坦:

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 中的协议提供了一种将静态类型引入鸭子类型代码的强大方法。它们允许我们在类型系统中指定接口,而不需要继承,保持鸭子类型的灵活性,同时增加静态类型检查的好处,

通过使用协议,您可以:

  1. 为您的代码定义清晰的接口
  2. 获得更好的 ide,(静态类型检查),支持并更早捕获错误
  3. 保持鸭子打字的灵活性
  4. 利用类型检查来检查您无法修改的类。

如果您想了解有关 python 中的协议和类型提示的更多信息,请查看有关类型模块的 python 官方文档,或探索 mypy 等高级静态类型检查工具。

快乐编码,愿你的鸭子总是因类型安全而嘎嘎叫!

您可以在这里找到更多我的内容,包括我的时事通讯

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1179

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

215

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2112

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

24

2026.01.19

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

0

2026.01.31

go语言 math包
go语言 math包

本专题整合了go语言math包相关内容,阅读专题下面的文章了解更多详细内容。

1

2026.01.31

go语言输入函数
go语言输入函数

本专题整合了go语言输入相关教程内容,阅读专题下面的文章了解更多详细内容。

1

2026.01.31

golang 循环遍历
golang 循环遍历

本专题整合了golang循环遍历相关教程,阅读专题下面的文章了解更多详细内容。

0

2026.01.31

Golang人工智能合集
Golang人工智能合集

本专题整合了Golang人工智能相关内容,阅读专题下面的文章了解更多详细内容。

1

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 22.4万人学习

Django 教程
Django 教程

共28课时 | 3.7万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号