当前位置: 首页 > article >正文

python小知识-typing注解你的程序

python小知识-typing注解你的程序

1. Typing的简介

typing 是 Python 的一个标准库,它提供了类型注解的支持,但并不会强制类型检查。类型注解在 Python 3.5 中引入,并在后续版本中得到了增强和扩展。typing 库允许开发者为变量、函数参数和返回值等提供预期的类型信息,这有助于代码的可读性、可维护性和文档化。此外,一些第三方工具(如 MyPy)可以使用这些类型注解进行静态类型检查。

2. Typing的基本使用

基本使用包括为变量、函数参数和返回值添加类型注解。

from typing import List, Dict, Union

# 变量类型注解
def greet(name: str) -> None:
    print(f"Hello, {name}!")

# 使用类型注解的变量
names: List[str] = ["Alice", "Bob", "Charlie"]

# 字典的类型注解
person_info: Dict[str, int] = {"name": "John", "age": 30}  # 注意:这里只是示例,实际中"name"应为str类型
# 正确的字典类型注解
person_info: Dict[str, Union[str, int]] = {"name": "John", "age": 30}  # 需要导入Union

可以看出第一个person_info并没有报错,只是注解的不对,可以正常使用。

Typing还提供了元组Tuple和可选类型Optional

Tuple 允许你为元组中的每个元素指定类型。

from typing import Tuple

# 定义一个包含两个整数的元组类型
IntPair = Tuple[int, int]

# 创建一个符合 IntPair 类型的元组
coordinate: IntPair = (10, 20)

# 函数返回元组
def get_coordinates() -> IntPair:
    return 1, 2

# 调用函数并访问返回的元组
x, y = get_coordinates()
print(f"Coordinates: ({x}, {y})")

Optional 表示一个值可以是某个类型或者 None

from typing import Optional

# 定义一个函数,其参数可能是字符串或None
def greet_optional(name: Optional[str] = None) -> None:
    if name is None:
        print("Hello, anonymous user!")
    else:
        print(f"Hello, {name}!")

# 调用函数,传入一个字符串
greet_optional("Alice")  # 输出: Hello, Alice!

# 调用函数,不传入任何参数(使用默认值None)
greet_optional()  # 输出: Hello, anonymous user!

3. 泛型的使用

泛型允许你编写灵活且可重用的代码,其中类型参数可以是任何类型。

在Python的typing库中,泛型(Generics)是一个非常重要的概念,它允许我们编写更加灵活和可重用的代码。泛型的主要意义和作用体现在以下几个方面:

提高代码复用性

泛型允许我们编写不依赖于具体类型的代码。通过定义类型参数(Type Parameters),我们可以创建可重用的类、函数和容器,这些结构能够处理任何数据类型。这样,我们就可以避免为每种类型重复编写相同的代码。

类型安全

虽然Python是一种动态类型语言,但在某些情况下,类型安全对于减少错误和提高代码质量非常重要。泛型提供了一种在编译时(或静态分析时)检查类型安全性的方法。通过使用泛型,我们可以确保传递给函数或类的参数类型是正确的,从而避免运行时错误。

提高代码可读性

泛型可以使代码更具描述性和可读性。通过明确指定泛型类型参数,我们可以使代码更清晰地表达其意图,这对于维护和理解代码非常有帮助。

容器类型抽象

泛型在容器类型(如列表、集合、字典等)的抽象中特别有用。通过使用泛型,我们可以定义能够处理任何数据类型的容器,而无需为每种类型单独编写代码。例如,我们可以定义一个泛型列表类,该类可以存储任何类型的元素,并提供统一的接口来操作这些元素。

泛型约束和类型推断

泛型还支持类型约束和类型推断。通过指定类型参数的上界或下界,我们可以限制可以传递给泛型结构的数据类型范围。此外,某些情况下,类型推断可以帮助我们自动推断出泛型类型参数的实际类型,从而简化代码编写。

下面是一个使用泛型实现栈的简单示例:

from typing import TypeVar, Generic, List

T = TypeVar('T')  # 声明一个类型变量T

class Stack(Generic[T]):
    def __init__(self):
        self.items: List[T] = []  # 泛型列表用于存储栈中的元素

    def push(self, item: T) -> None:
        self.items.append(item)

    def pop(self) -> T:
        if not self.items:
            raise IndexError("pop from an empty stack")
        return self.items.pop()

# 使用整型栈
int_stack = Stack[int]()
int_stack.push(1)
int_stack.push(2)
print(int_stack.pop())  # 输出: 2

# 使用字符串栈
str_stack = Stack[str]()
str_stack.push("hello")
str_stack.push("world")
print(str_stack.pop())  # 输出: world

在上面的示例中,我们定义了一个泛型栈类Stack,它使用类型变量T来表示栈中元素的类型。通过指定不同的类型参数(如intstr),我们可以创建不同类型的栈,并享受泛型带来的代码复用性和类型安全性。

4. 自定义新的类型

虽然 typing 库已经提供了许多内置的类型注解,但你也可以使用 NewType 来创建新的类型别名,这些别名在类型检查时会视为不同的类型。

from typing import NewType

UserId = NewType('UserId', int)

def get_user_name(user_id: UserId) -> str:
    # 假设有一个根据用户ID获取用户名的函数
    pass

# 正确使用
user_id: UserId = UserId(123)
get_user_name(user_id)

# 错误使用(类型不匹配)
get_user_name(123)  # 如果使用MyPy等类型检查工具,这里会报错

5. 总结

typing 库是 Python 中非常重要的一个库,它提供了类型注解的支持,使得代码更加清晰、易于理解和维护。通过使用 typing 库,我们可以为变量、函数参数和返回值等添加类型信息,从而提高代码的可读性和可维护性。此外,泛型的使用可以让我们编写更加灵活和可重用的代码。最后,通过自定义新的类型别名,我们可以进一步细化类型信息,提高代码的类型安全性。


http://www.kler.cn/a/528336.html

相关文章:

  • 【VM】VirtualBox安装CentOS8虚拟机
  • BurpSuite抓包与HTTP基础
  • AutoDL 云服务器:xfce4 远程桌面 终端乱码 + 谷歌浏览器
  • [Collection与数据结构] B树与B+树
  • CUDA学习-内存访问
  • 深入理解MySQL 的 索引
  • Flutter开发环境配置
  • 【Uniapp-Vue3】解决uni-popup弹窗在安全区显示透明问题
  • Linux——ext2文件系统(一)
  • 使用 Redis Streams 实现高性能消息队列
  • 2025 AI行业变革:从DeepSeek V3到o3-mini的技术演进
  • 蓝桥杯刷题DAY2:二维前缀和 一维前缀和 差分数组
  • leetcode 2563. 统计公平数对的数目
  • x86-64数据传输指令
  • 【Pytorch和Keras】使用transformer库进行图像分类
  • Python-基于PyQt5,pdf2docx,pathlib的PDF转Word工具
  • 海外问卷调查,最常用到的渠道查有什么特殊之处
  • XML DOM - 导航节点
  • CSDN图片加载不出来问题解决
  • webrtc peerconnection_client peerconnection_server 连接失败问题解决 win10 win11
  • 使用UpdateCursor删除行
  • 学习ArkTS语言
  • 开源2 + 1链动模式AI智能名片S2B2C商城小程序视角下从产品经营到会员经营的转型探究
  • react中useEffect的使用
  • 【数据结构】_C语言实现带头双向循环链表
  • 安装anaconda3 后 电脑如何单独运行python,python还需要独立安装吗?