Python 3.14 有什么新变化

编者:

Hugo van Kemenade

本文介绍了 Python 3.14 相比 3.13 的新增特性。

完整的详情可参阅 更新日志

参见

PEP 745 -- Python 3.14 发布计划

备注

预发布版用户应当了解到此文档目前处于草稿状态。 它将随着 Python 3.14 的发布进程不断更新,因此即使已经阅读过较早的版本也仍然值得再次查看。

摘要 -- 发布关键要点

Python 3.14 将成为 Python 编程语言的最新稳定版本,包含对语言本身、实现细节及标准库的一系列更新。

对于实现的最大变化包括模板字符串 (PEP 750),标注的迟延求值 (PEP 649),以及一种使用尾调用的新解释器类型。

标准库的变化包括新增 annotationlib 模块用于内省和包裹标注 (PEP 749),新增 compression.zstd 模块用于 Zstandard 支持 (PEP 784),以及 REPL 中的语法高亮,以及常规的弃用和移除,还有用户友好度和正确性方面的改进。

不兼容的更改

在 macOS 和 Windows 以外的平台上,用于 multiprocessingProcessPoolExecutor 的默认 start 方法fork 改为 forkserver

请参阅 (1)(2) 了解详情。

如果你在 multiprocessingconcurrent.futures 中遇到 NameError 或 pickle 错误,请参阅 forkserver 的限制

解释器会在保证安全的情况下避免某些引用计数修改。 这可能导致 sys.getrefcount()Py_REFCNT() 返回相比之前 Python 版本有所不同的值。 请参阅 下文 了解详情。

新的特性

PEP 779: 自由线程 Python 已获官方支持

自由线程的 Python 构建版现在已被支持而不再是实验性的。 这是自由线程 Python 获得官方支持但仍为可选项的阶段 II 的开始。

我们相信该项目走在正确的路径上,我们赞赏各位持续努力的贡献使得自由线程特性已准备好在 Python 社区获得更广泛的接受。

随着这些建议和这个 PEP 被接受,我们 Python 开发者社区应当广泛宣传自由线程现在已成为受支持的 Python 构建选项并在未来继续发展,它不会在没有适当的弃用计划的情况下被移除。

任何过渡到第三阶段(将自由线程作为Python的默认或唯一构建)的决定仍未决定,这取决于CPython本身和社区内部的许多因素。这个决定是为了将来。

参见

PEP 779 及其 接受.

PEP 734: 添加多解释器到标准库

CPython 运行时支持在同一个进程中同时运行多个 Python 的副本并且已经这样做了 20 年以上。 每个这样的单独副本被称为“解释器”。 不过,该特性过去只能通过 C-API 来使用。

通过新增的 concurrent.interpreters 模块,这一限制已在 3.14 发布版中被移除。

使用多解释器的做法值得被考虑至少有两个重要的理由:

  • 它们支持(对 Python 来说)全新的、用户友好的并发模型

  • 真正的多核心并行

在某些应用场景中,软件中的并发机制能够在宏观层面上提高效率并简化软件设计。但与此同时,实现和维护除最简单形式之外的并发逻辑,通常对人脑来说是一项挑战。这一点尤其适用于普通线程(例如:threading),其中所有线程之间共享全部内存。

通过使用多个隔离的解释器,您可以利用一类在其他编程语言(如 Smalltalk、Erlang、Haskell 和 Go)中已被证明成功的并发模型,例如 CSP 或演员模型。可以将多个解释器视为线程,但它们之间的资源共享是可选的。

关于多核并行性:从 3.12 版本开始,各个解释器之间已实现足够的隔离,从而可以并行使用。(参见 PEP 684。)这使得 Python 能够解锁一系列此前受 GIL 限制的 CPU 密集型应用场景。

使用多个解释器在许多方面与 multiprocessing 类似,因为它们都提供了相互隔离的逻辑“进程”,默认情况下不共享任何资源,并且可以并行运行。然而,在使用多个解释器时,应用程序将占用更少的系统资源,并能以更高的效率运行(因为它仍处于同一个进程内)。 可以将多个解释器看作是:拥有进程级别的隔离性,同时具备线程级别的执行效率。

尽管该特性已经存在数十年,但由于认知度较低且缺乏标准库模块的支持,多个解释器并未被广泛使用。因此,目前它仍存在一些显著的限制。不过,随着这一特性终于开始走向主流,这些限制将得到显著改善。

当前限制:

  • 启动每个解释器尚未经过优化。

  • 每个解释器目前占用的内存比实际需要的更多(我们下一步将重点推进解释器之间的内部共享,以优化这一问题)。

  • 解释器之间真正实现对象或其他数据共享的选项还很有限(除了 memoryview)。

  • PyPI 上的许多扩展模块目前尚不兼容多个解释器(标准库中的扩展模块 兼容的)。

  • 目前,针对使用多个隔离解释器编写应用程序的方法,对大多数 Python 用户来说仍较为陌生

这些限制的影响将取决于未来 CPython 的改进程度、解释器的使用方式,以及社区通过 PyPI 包所解决的问题。根据具体的应用场景,这些限制可能并不会造成太大影响,因此不妨尝试一下!

此外,未来的 CPython 版本将进一步减少甚至消除相关开销,并提供一些目前在 PyPI 上不太合适的工具。在此期间,大多数限制也可以通过扩展模块来解决,这意味着 PyPI 包可以填补 3.14 乃至回溯到 3.12(此时解释器终于实现真正隔离并停止共享 GIL)之间的功能缺口。同样地,我们预计 PyPI 上将逐渐出现基于多解释器构建的高级抽象库。

关于扩展模块,目前正在进行一些 PyPI 项目的更新,以及对 Cython、pybind11、nanobind 和 PyO3 等工具的支持工作。有关隔离扩展模块的具体步骤,可参考 隔离扩展模块。模块的隔离与支持 自由线程 所需的工作有大量重叠,因此社区在该领域的持续努力将有助于加快对多个解释器的支持进程。

在 3.14 中还增加了:concurrent.futures.InterpreterPoolExecutor

参见

PEP 734.

PEP 750:模板字符串

模板字符串(t-字符串)是f-字符串的通用化,使用 t 代替 f 前缀。 与前者求值为 str 不同,t-字符串将求值为新增的 string.templatelib.Template 类型:

from string.templatelib import Template

name = "World"
template: Template = t"Hello {name}"

随后 template 可以被用于操作该模板的结构以产生 str 或字符串型结果的函数。 例如,对输入进行无害化:

evil = "<script>alert('evil')</script>"
template = t"<p>{evil}</p>"
assert html(template) == "<p>&lt;script&gt;alert('evil')&lt;/script&gt;</p>"

再比如,根据数据生成 HTML 属性:

attributes = {"src": "shrubbery.jpg", "alt": "looks nice"}
template = t"<img {attributes}>"
assert html(template) == '<img src="shrubbery.jpg" alt="looks nice" />'

与使用 f-string 相比,html 函数能够访问包含原始信息的模板属性:静态字符串、插值表达式,以及来自原始作用域的值。与现有的模板系统不同,t-string 是基于广为人知的 f-string 语法和规则构建的。因此,模板系统能够更好地受益于 Python 的工具生态,因为它们在语言特性、语法结构、作用域规则等方面都更加贴近 Python 本身。

编写模板处理程序非常简单:

from string.templatelib import Template, Interpolation

def lower_upper(template: Template) -> str:
    """Render static parts lowercased and interpolations uppercased."""
    parts: list[str] = []
    for item in template:
        if isinstance(item, Interpolation):
            parts.append(str(item.value).upper())
        else:
            parts.append(item.lower())
    return "".join(parts)

name = "world"
assert lower_upper(t"HELLO {name}") == "hello WORLD"

使用此特性,开发者可以编写模板系统来对 SQL 做无害化处理,执行安全的 shell 操作,改进日志记录,处理 Web 开发中的现代概念(HTML、CSS 等等),以及实现轻量的、定制的业务 DSL。

(由 Jim Baker、Guido van Rossum、Paul Everitt、Koudai Aono、Lysandros Nikolaou、Dave Peck、Adam Turner、Jelle Zijlstra、Bénédikt Tran 和 Pablo Galindo Salgado 在 gh-132661 中贡献。)

参见

PEP 750

PEP 768:Cpython 的安全外部调试器接口

PEP 768 引入了一个零开销的调试接口,允许调试器和性能分析工具安全地附加到正在运行的 Python 进程上。这是对 Python 调试能力的重要增强,使调试器可以避免使用不安全的替代方案。有关该特性如何用于实现新版 pdb 模块的远程附加功能,请参见 下面

新的接口在不修改解释器正常执行路径或增加运行时开销的前提下,提供了安全的调试代码附加点。这使得工具能够在不停止或重启 Python 应用程序的情况下,实时地对其进行检查和交互——这对于高可用性系统和生产环境至关重要。

为方便起见,CPython 通过 sys 模块中的 sys.remote_exec() 函数实现该接口:

sys.remote_exec(pid, script_path)

此函数允许发送 Python 代码以便在目标进程中的下一个安全执行点上执行。 不过,工具作者也可以直接实现在 PEP 中描述的协议,它详细讲解了用于安全附加到运行进程的底层机制。

以下是一个检查运行中 Python 进程内对象类型的简单示例:

import os
import sys
import tempfile

# 创建一个临时脚本
with tempfile.NamedTemporaryFile(mode='w', suffix='.py', delete=False) as f:
    script_path = f.name
    f.write(f"import my_debugger; my_debugger.connect({os.getpid()})")
try:
    # 在进程 PID 1234 中执行:
    print("Behold! An offering:")
    sys.remote_exec(1234, script_path)
finally:
    os.unlink(script_path)

该调试接口在设计时已充分考虑安全性,并包含多种访问控制机制:

关键实现细节在于:该接口复用了解释器现有的求值循环和安全点机制,既确保正常执行时零开销,又为外部进程提供了可靠的调试操作协调方式。

(由 Pablo Galindo Salgado,Matt Wozniski 和 Ivona Stojanovic 在 gh-131591 中贡献)

参见

PEP 768

PEP 784: 添加 Zstandard 到标准库

新推出的 compression 包包含以下模块:compression.lzmacompression.bz2compression.gzipcompression.zlib,它们分别重新导出了 lzmabz2gzipzlib 模块。今后,compression 下的新导入名称将成为这些压缩模块的标准导入方式。不过,现有的模块名称尚未被弃用。任何对现有压缩模块的弃用或移除都不会早于 3.14 版本发布后的五年。

新引入的 compression.zstd 模块通过绑定 Meta的zstd库 提供了Zstandard格式的压缩和解压API。Zstandard是一种被广泛采用、高效且快速的压缩格式。除了 compression.zstd 中引入的API外,对Zstandard压缩归档文件的读写支持也已添加到 tarfilezipfileshutil 模块中。

下面是一个使用新模块压缩数据的示例:

from compression import zstd
import math

data = str(math.pi).encode() * 20

compressed = zstd.compress(data)

ratio = len(compressed) / len(data)
print(f"达到的压缩比为 {ratio}")

可以看出,该API与 lzmabz2 模块的API类似。

(由 Emma Harper Smith、Adam Turner、Gregory P. Smith、Tomas Roun、Victor Stinner 和 Rogdham 在 gh-132983 中贡献。)

参见

PEP 784

使用PDB远程附加到正在运行的Python进程

pdb`模块现在支持通过新的 `-p PID`` 命令行选项远程附加到正在运行的Python进程:

python -m pdb -p 1234

该操作将连接到指定PID的Python进程,并允许您进行交互式调试。请注意,由于Python解释器的工作原理,当附加到阻塞在系统调用或等待I/O的远程进程时,调试功能只有在执行下一条字节码指令或进程收到信号时才会生效。

该功能使用 PEP 768sys.remote_exec() 函数来附加到远程进程,并向其发送PDB命令。

(由 Matt Wozniski 和 Pablo Galindo 在 gh-131591 中贡献。)

参见

PEP 768

PEP 758 – 允许不带括号的 except 和 except* 表达式

exceptexcept* 表达式现在允许在存在多个异常类型且未使用 as 子句时省略括号。例如,以下表达式现在有效:

try:
    connect_to_server()
except TimeoutError, ConnectionRefusedError:
    print("遇到网络问题。")

 # 同样适用于 except* (针对异常组):

try:
    connect_to_server()
except* TimeoutError, ConnectionRefusedError:
    print("遇到网络问题。")

请参阅 PEP 758 获取更多细节。

(由 Pablo Galindo 和 Brett Cannon 在 gh-131831 中贡献。)

参见

PEP 758

PEP 649 和 749:注解的延迟求值

在函数、类和模块上的 注解 不再被立即求值。相反,注解会被存储在专用的 注解函数 中,仅在必要时进行求值(除非使用了 from __future__ import annotations)。这一特性在 PEP 649PEP 749 中进行了规范。

这一改进旨在使 Python 中的注解在大多数情况下性能更高、更易用。定义注解的运行时开销被降至最低,同时仍可在运行时内省注解。如果注解包含前向引用,也不再需要将其包裹在字符串中。

新引入的 annotationlib 模块提供了检查延迟注解的工具。注解可以通过以下格式进行求值:VALUE 格式(将注解求值为运行时值,类似于早期Python版本的行为)、FORWARDREF 格式(用特殊标记替换未定义名称)、STRING 格式(以字符串形式返回注解)。

以下示例展示了这些格式的具体行为:

>>> from annotationlib import get_annotations, Format
>>> def func(arg: Undefined):
...     pass
>>> get_annotations(func, format=Format.VALUE)
Traceback (most recent call last):
  ...
NameError: name 'Undefined' is not defined
>>> get_annotations(func, format=Format.FORWARDREF)
{'arg': ForwardRef('Undefined', owner=<function func at 0x...>)}
>>> get_annotations(func, format=Format.STRING)
{'arg': 'Undefined'}

注解代码的影响

如果在代码中定义注解(例如用于静态类型检查器),那么这一变更可能不会产生影响:你可以继续保持与之前Python版本相同的注解书写方式。

你很可能可以移除注解中的引号字符串(这些通常用于前向引用)。同样地,如果使用 from __future__ import annotations 来避免在注解中书写字符串,当你仅支持 Python 3.14 及更新版本时,很可能可以移除该导入。不过,如果依赖读取注解的第三方库,这些库可能需要相应修改以支持无引号注解,才能正常工作。

访问 __annotations__ 的影响

如果代码会读取对象的 __annotations__ 属性,你可能需要进行调整以支持依赖注解延迟求值的代码。例如,可以像 dataclasses 模块现在所做的那样,使用 annotationlib.get_annotations() 函数并指定 FORWARDREF 格式。

外部包 typing_extensions 提供了 annotationlib 模块部分功能的向后兼容实现,包括 Format 枚举和 get_annotations() 函数。这些可用于编写跨版本代码,以利用 Python 3.14 中的新行为。

from __future__ import annotations

在 Python 3.7 中,PEP 563 引入了 from __future__ import annotations 指令,该指令会将所有注解转换为字符串。此指令现已被视为弃用,预计将在未来的 Python 版本中移除。不过,在 Python 3.13(最后一个不支持注解延迟求值的 Python 版本)于 2029 年结束生命周期之前,这一移除操作不会实施。在 Python 3.14 中,使用 from __future__ import annotations 的代码行为保持不变。

(由 Jelle Zijlstra 在 gh-119180 中贡献;PEP 649 由 Larry Hastings 编写。)

参见

PEP 649PEP 749

改进的错误消息

  • 解释器现在能在检测到Python关键字拼写错误时提供有用的建议。当遇到与Python关键字高度相似的单词时,解释器将在错误信息中建议正确的关键字。该功能可帮助程序员快速识别和修复常见的输入错误。例如:

    >>> whille True:
    ...     pass
    Traceback (most recent call last):
      File "<stdin>", line 1
        whille True:
        ^^^^^^
    SyntaxError: invalid syntax. Did you mean 'while'?
    
    >>> asynch def fetch_data():
    ...     pass
    Traceback (most recent call last):
      File "<stdin>", line 1
        asynch def fetch_data():
        ^^^^^^
    SyntaxError: invalid syntax. Did you mean 'async'?
    
    >>> async def foo():
    ...     awaid fetch_data()
    Traceback (most recent call last):
      File "<stdin>", line 2
        awaid fetch_data()
        ^^^^^
    SyntaxError: invalid syntax. Did you mean 'await'?
    
    >>> raisee ValueError("Error")
    Traceback (most recent call last):
      File "<stdin>", line 1
        raisee ValueError("Error")
        ^^^^^^
    SyntaxError: invalid syntax. Did you mean 'raise'?
    

    虽然该功能主要针对最常见的拼写错误情况,但某些变体的拼写错误仍可能导致常规语法错误。(由 Pablo Galindo 在 gh-132449 中贡献。)

  • 当一个解包赋值因变量数量不匹配而失败时,错误消息现在会在更多情况下显示实际接收到的数值数量。(由 Tushar Sadhwani 在 gh-122239 中贡献。)

    >>> x, y, z = 1, 2, 3, 4
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
        x, y, z = 1, 2, 3, 4
        ^^^^^^^
    ValueError: too many values to unpack (expected 3, got 4)
    
  • 跟在 else 块后的 elif 语句现在会触发特定的错误提示。(由 Steele Farnsworth 在 gh-129902 中贡献。)

    >>> if who == "me":
    ...     print("It's me!")
    ... else:
    ...     print("It's not me!")
    ... elif who is None:
    ...     print("Who is it?")
    File "<stdin>", line 5
      elif who is None:
      ^^^^
    SyntaxError: 'elif' block follows an 'else' block
    
  • 当以下语句(passdelreturnyieldraisebreakcontinueassertimportfrom)被传递到 else 后的 条件表达式,或者 passbreakcontinue 之一被传递到 if 之前时,错误消息会明确标出需要 expression 的位置。(由 Sergey Miryanov 在 gh-129515 中贡献。)

    >>> x = 1 if True else pass
    Traceback (most recent call last):
      File "<string>", line 1
        x = 1 if True else pass
                           ^^^^
    SyntaxError: expected expression after 'else', but statement is given
    
    >>> x = continue if True else break
    Traceback (most recent call last):
      File "<string>", line 1
        x = continue if True else break
            ^^^^^^^^
    SyntaxError: expected expression before 'if', but statement is given
    
  • 当检测到未正确闭合的字符串时,错误消息会提示该字符串可能是字符串的一部分。(由 Pablo Galindo 在 gh-88535 中贡献。)

    >>> "The interesting object "The important object" is very important"
    Traceback (most recent call last):
    SyntaxError: invalid syntax. Is this intended to be part of the string?
    
  • 当字符串前缀不兼容时,错误提示现在会明确显示哪些前缀存在冲突。(由 Nikita Sobolev 在 gh-133197 中贡献。)

    >>> ub'abc'
      File "<python-input-0>", line 1
        ub'abc'
        ^^
    SyntaxError: 'u' and 'b' prefixes are incompatible
    
  • 在以下场景中使用不兼容目标的 as 语句时,错误提示已得到改进:

    • 导入:import ... as ...

    • From 导入:from ... import ... as ...

    • Except 处理器:except ... as ...

    • 模式匹配 case 语句:case ... as ...

    (由 Nikita Sobolev 在 gh-123539gh-123562gh-123440 中贡献。)

    >>> import ast as arr[0]
      File "<python-input-1>", line 1
        import ast as arr[0]
                      ^^^^^^
    SyntaxError: cannot use subscript as import target
    
  • 尝试向 dictset 添加不可哈希类型的实例时,错误提示信息已改进。(由 CF Bolz-Tereick 和 Victor Stinner 在 gh-132828 中贡献。)

    >>> s = set()
    >>> s.add({'pages': 12, 'grade': 'A'})
    Traceback (most recent call last):
      File "<python-input-1>", line 1, in <module>
        s.add({'pages': 12, 'grade': 'A'})
        ~~~~~^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
    TypeError: cannot use 'dict' as a set element (unhashable type: 'dict')
    >>> d = {}
    >>> l = [1, 2, 3]
    >>> d[l] = 12
    Traceback (most recent call last):
      File "<python-input-4>", line 1, in <module>
        d[l] = 12
        ~^^^
    TypeError: cannot use 'list' as a dict key (unhashable type: 'list')
    

PEP 741:Python 配置 C API

新增 PyInitConfig C API 用于配置 Python 初始化过程,无需依赖 C 结构体,同时支持未来进行 ABI 兼容性变更。

通过添加 PyInitConfig_AddModule() 函数来完成 PEP 587 规范中定义的 PyConfig C API,该函数可用于添加内置扩展模块;该特性此前被称为“inittab”。

新增 PyConfig_Get()PyConfig_Set() 函数,用于获取和设置当前运行时配置。

PEP 587《Python 初始化配置》统一了所有配置 Python 初始化的方式。该PEP进一步将 Python 预初始化配置与初始化配置整合至单一API中。此外,为进一步简化API设计,本PEP仅提供"嵌入式Python"这一种选择,而非PEP 587中的"Python"和"隔离模式"双选项方案。

较低层级的 PEP 587 PyConfig API 仍保留可用,适用于需要与 CPython 实现细节保持较高耦合度的使用场景(例如模拟 CPython CLI 的完整功能,包括其配置机制)。

(由 Victor Stinner 在 gh-107954 中贡献。)

参见

PEP 741

asyncio 内省能力

新增了一个通过异步任务检查运行中Python进程的命令行接口,可通过以下方式使用:

python -m asyncio ps PID

该工具会检查指定的进程ID(PID),并显示当前正在运行的asyncio任务信息。其输出包含一个任务表格:以扁平列表形式展示所有任务及其名称、协程堆栈信息,以及各任务正在等待的其他任务。

python -m asyncio pstree PID

该工具获取相同的信息,但会以可视化异步调用树的形式呈现,采用层级结构展示协程关系。此命令特别适用于调试长时间运行或卡死的异步程序,能帮助开发者快速定位程序阻塞位置、识别待处理任务,以及理清协程间的调用链关系。

例如给定以下代码:

import asyncio

async def play(track):
    await asyncio.sleep(5)
    print(f"🎵 Finished: {track}")

async def album(name, tracks):
    async with asyncio.TaskGroup() as tg:
        for track in tracks:
            tg.create_task(play(track), name=track)

async def main():
    async with asyncio.TaskGroup() as tg:
        tg.create_task(
          album("Sundowning", ["TNDNBTG", "Levitate"]), name="Sundowning")
        tg.create_task(
          album("TMBTE", ["DYWTYLM", "Aqua Regia"]), name="TMBTE")

if __name__ == "__main__":
    asyncio.run(main())

在运行中的进程上执行该新工具将生成如下表格:

python -m asyncio ps 12345

tid        task id              task name            coroutine stack                                    awaiter chain                                      awaiter name    awaiter id
------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
1935500    0x7fc930c18050       Task-1               TaskGroup._aexit -> TaskGroup.__aexit__ -> main                                                                       0x0
1935500    0x7fc930c18230       Sundowning           TaskGroup._aexit -> TaskGroup.__aexit__ -> album   TaskGroup._aexit -> TaskGroup.__aexit__ -> main    Task-1          0x7fc930c18050
1935500    0x7fc93173fa50       TMBTE                TaskGroup._aexit -> TaskGroup.__aexit__ -> album   TaskGroup._aexit -> TaskGroup.__aexit__ -> main    Task-1          0x7fc930c18050
1935500    0x7fc93173fdf0       TNDNBTG              sleep -> play                                      TaskGroup._aexit -> TaskGroup.__aexit__ -> album   Sundowning      0x7fc930c18230
1935500    0x7fc930d32510       Levitate             sleep -> play                                      TaskGroup._aexit -> TaskGroup.__aexit__ -> album   Sundowning      0x7fc930c18230
1935500    0x7fc930d32890       DYWTYLM              sleep -> play                                      TaskGroup._aexit -> TaskGroup.__aexit__ -> album   TMBTE           0x7fc93173fa50
1935500    0x7fc93161ec30       Aqua Regia           sleep -> play                                      TaskGroup._aexit -> TaskGroup.__aexit__ -> album   TMBTE           0x7fc93173fa50

或生成如下树状结构:

python -m asyncio pstree 12345

└── (T) Task-1
    └──  main example.py:13
        └──  TaskGroup.__aexit__ Lib/asyncio/taskgroups.py:72
            └──  TaskGroup._aexit Lib/asyncio/taskgroups.py:121
                ├── (T) Sundowning
                   └──  album example.py:8
                       └──  TaskGroup.__aexit__ Lib/asyncio/taskgroups.py:72
                           └──  TaskGroup._aexit Lib/asyncio/taskgroups.py:121
                               ├── (T) TNDNBTG
                                  └──  play example.py:4
                                      └──  sleep Lib/asyncio/tasks.py:702
                               └── (T) Levitate
                                   └──  play example.py:4
                                       └──  sleep Lib/asyncio/tasks.py:702
                └── (T) TMBTE
                    └──  album example.py:8
                        └──  TaskGroup.__aexit__ Lib/asyncio/taskgroups.py:72
                            └──  TaskGroup._aexit Lib/asyncio/taskgroups.py:121
                                ├── (T) DYWTYLM
                                   └──  play example.py:4
                                       └──  sleep Lib/asyncio/tasks.py:702
                                └── (T) Aqua Regia
                                    └──  play example.py:4
                                        └──  sleep Lib/asyncio/tasks.py:702

当检测到异步等待图中存在循环引用(可能表明存在编程问题)时,该工具将报错并列出阻碍树形结构构建的循环路径:

python -m asyncio pstree 12345

ERROR: await-graph contains cycles - cannot print a tree!

cycle: Task-2  Task-3  Task-2

(由 Pablo Galindo、 Łukasz Langa、 Yury Selivanov 和 Marta Gomez Macias 在 gh-91048 中贡献。)

一种新型的解释器

CPython 新增了一种全新的解释器类型。它通过在实现各个 Python 操作码的小型 C 函数之间使用尾调用来运行,而不是使用单个大型 C case 语句。对于某些较新的编译器,这种解释器能带来显著的性能提升。我们在测试机器上的初步数据显示,Python 代码的运行速度最高可提升 30%,在不同平台和架构上运行 pyperformance 基准测试的几何平均提速为 3% 到 5%。作为对比的基线是使用 Clang 19 编译、未启用该新解释器的 Python 3.14。

此解释器目前仅适用于 x86-64 和 AArch64 架构上的 Clang 19 及更高版本。不过,我们预计 GCC 的未来版本也将支持此功能。

此功能目前为可选功能。 我们强烈建议在新解释器中启用配置文件引导优化,因为这是我们测试过的唯一配置,可以验证其性能提升。 有关如何构建 Python 的更多信息,请参阅 --with-tail-call-interp

备注

这不应与Python函数的 尾调用优化 相混淆,该特性目前在CPython中尚未实现。

这一新型解释器属于CPython解释器的内部实现细节,完全不会改变Python程序的可见行为。它能提升程序性能,但不会引发任何其他变更。

注意

本节原先报告的几何平均加速比为 9-15%,但这一数字现已谨慎地下调为 3% 到 5%。 尽管我们预计实际性能提升可能高于这一范围,但由于在 Clang/LLVM 19 中发现了一个`编译器漏洞 <https://github.com/llvm/llvm-project/issues/106846>`_,我们对性能估算变得更加保守。该漏洞会导致常规解释器变慢,我们在早期并未意识到这一点,导致报告的性能数据不准确。 对此我们深感抱歉,之前的数据实际上仅对 LLVM v19.1.x 和 v20.1.0 有效。 目前,这个漏洞已经在 LLVM v20.1.1 和即将发布的 v21.1 中被修复,但在 LLVM v19.1.x 和 v20.1.0 中将不会得到修复。 因此,使用这些 LLVM 版本进行的基准测试可能会生成不准确的数据。 (感谢 Nelson Elhage 揭示了这个问题。)

(由 Ken Jin 在 gh-128563 中贡献,CPython实现方案融合了Mark Shannon、Garrett Gu、Haoran Xu和Josh Haberman的设计理念。)

自由线程模式

自由线程模式(PEP 703,最初在3.13版本引入)已得到显著改进。PEP 703中描述的实现方案已完成,包括C API变更,且解释器中的临时解决方案已被更持久的方案取代。专业化自适应解释器(PEP 659)现已在自由线程模式中启用,结合众多其他优化措施,其性能得到大幅提升。当前自由线程模式对单线程代码的性能影响已降至约5-10%,具体取决于平台和使用的C编译器。

此项工作由众多贡献者共同完成:Sam Gross、Matt Page、Neil Schemenauer、Thomas Wouters、Donghee Na、Kirill Podoprigora、Ken Jin、Itamar Oren、Brett Simmers、Dino Viehland、Nathan Goldbaum、Ralf Gommers、Lysandros Nikolaou、Kumar Aditya、Edgar Margffoy 以及其他很多人。

部分贡献者受雇于Meta,该公司持续为本项目提供了大量工程技术资源支持。

自3.14版本起,在Windows平台上为CPython的自由线程构建编译扩展模块时,构建后端现在需要显式指定预处理器变量 Py_GIL_DISABLED,因为该变量将不再由C编译器自动确定。对于运行中的解释器,可通过 sysconfig.get_config_var() 查询编译时使用的配置。

新增了一个旗标 context_aware_warnings。 这个旗标在自由线程构建版中默认为真值而在启用了 GIL 的构建版中为假值。 如果该旗标为真值则 warnings.catch_warnings 上下文管理器会使用一个上下文变量作为警告过滤器。 这使得该上下文管理器在多线程或异步任务场景下具有可预测的行为。

新增了一个标志位 thread_inherit_context。该标志在自由线程构建中默认为真(True),在启用GIL的构建中默认为假(False)。当标志为真时,通过 threading.Thread 创建的线程会继承 start() 调用者的 Context() 副本。最重要的是,这使得由 catch_warnings 建立的警告过滤上下文能够被该上下文中启动的线程(或asyncio任务)"继承"。该标志还会影响其他使用上下文变量的模块,例如 decimal 上下文管理器。

PyREPL 的语法高亮功能

默认的 interactive 式 shell 现已支持实时 Python 语法高亮。 除非设置了 PYTHON_BASIC_REPL 环境变量或使用了任何禁用颜色的环境变量,否则该功能默认启用。 详情请参阅 控制颜色

语法高亮的默认配色方案追求高对比度,并仅使用4位VGA标准ANSI颜色代码以确保最大兼容性。可通过实验性API _colorize.set_theme() 自定义主题,该API既支持交互式调用,也可在 PYTHONSTARTUP 脚本中使用。

(由 Łukasz Langa 在 gh-131507 中贡献。)

实验性即时编译器(JIT)的二进制发布版本

官方 macOS 和 Windows 发布版二进制文件现已包含 实验性 即时编译(JIT)器。虽然不建议在生产环境中使用,但可通过设置环境变量 PYTHON_JIT=1 进行测试。下游源码构建和再分发方可使用配置选项 --enable-experimental-jit=yes-off 实现类似行为。

该JIT编译器尚处于早期开发阶段,性能表现存在波动:启用后根据工作负载不同,可能产生10%的性能下降至20%的性能提升。为便于测试评估,sys._jit 命名空间提供了一组内省函数:sys._jit.is_available() 用于检测当前可执行文件是否支持JIT编译,而 sys._jit.is_available() 则可判断当前进程是否已启用JIT编译功能。

当前最显著的功能缺失是原生调试器(如 gdb)和性能分析工具(如 perf)无法展开JIT调用栈(而Python原生调试器如 pdb 和性能分析器如 profile 仍可无需修改直接使用)。此外,自由线程构建版本暂不支持JIT编译功能。

如遇任何错误或严重性能退化问题,请务必提交报告!

参见

PEP 744

并发安全的警告控制

warnings.catch_warnings 上下文管理器现在支持通过上下文变量处理警告过滤器,该功能需通过设置 context_aware_warnings 标志启用(可使用 -X 命令行选项或环境变量)。当结合多线程或异步任务使用 catch_warnings 时,此举能提供可预测的警告控制。该标志在自由线程构建中默认为真(True),在启用GIL的构建中默认为假(False)。

(由 Neil Schemenauer 和 Kumar Aditya 在 gh-130010 中贡献。)

增量式垃圾回收

循环垃圾回收器现在是增量式的。 这意味着针对较大的堆的最大暂停时间将减少一个或更多的数量级。

现在只有两个代:年轻代和老年代。当不直接调用 gc.collect() 时,GC的调用频率会稍低。当被调用时,它回收年轻代和老年代的一部分,而不是回收一个或多个代。

gc.collect() 的行为略有变化:

  • gc.collect(1): 执行增量垃圾回收,而不是回收第 1 代。

  • 其他对 gc.collect() 的调用保持不变。

(由 Mark Shannon 在 gh-108362 中贡献。)

平台支持

其他语言特性修改

  • 默认的 interactive shell 现已支持导入自动补全功能。 具体表现为:输入 import foo 后按下 <tab> 键会建议以 foo 开头的模块;类似地,输入 from foo import b 则会建议以 b 开头的 foo 子模块。 需注意当前暂不支持模块属性的自动补全。 (由 Tomas Roun 在 gh-69605 中贡献。)

  • 内置函数 map() 现在新增了一个可选的仅限关键字参数 strict (与 zip() 类似),用于校验所有可迭代对象长度是否一致。(由 Wannes Boeykens 在 gh-119793 中贡献。)

  • 现在即使代码被 -O 命令行选项优化移除,仍能检测出 await 和异步推导式的错误用法。例如,执行 python -O -c 'assert await 1' 现在会触发 SyntaxError。(由 Jelle Zijlstra 在 gh-121637 中贡献。)

  • 现在即使代码被 -O 命令行选项优化移除,对 __debug__ 的写入操作仍会被检测到。例如,执行 python -O -c 'assert (__debug__ := 1)' 现在会触发 SyntaxError。(由 Irit Katriel 在 gh-122245 中贡献。)

  • 新增类方法 float.from_number()complex.from_number(),用于将数值分别转换为 floatcomplex 类型。若参数为字符串则会引发错误。(由 Serhiy Storchaka 在 gh-84978 中贡献。)

  • 实现了符合C99以来C标准规范的实数与复数混合运算规则。(由 Sergey B Kirpichev 在 gh-69639 中贡献。)

  • 现在Windows平台已支持所有Windows代码页作为"cpXXX"编解码器使用。(由 Serhiy Storchaka 在 gh-123803 中贡献。)

  • super 对象现在支持 序列化复制 操作。(由 Serhiy Storchaka 在 gh-125767 中贡献。)

  • memoryview 类型现已支持下标,使其成为 generic type。 (由 Brian Schubert 在 gh-126012 中贡献。)

  • 在新式字符串格式化(通过 format()f-字符串 实现)的浮点数表示类型中,现支持使用下划线和逗号作为小数部分的千位分隔符。(由 Sergey B Kirpichev 在 gh-87790 中贡献。)

  • bytes.fromhex()bytearray.fromhex() 方法现在支持 ASCII 格式的 bytes字节型对象 作为输入参数。(由 Daniel Pope 在 gh-129349 中贡献。)

  • 正则表达式 中,现支持将 \z 作为 \Z 的同义符使用。与行为存在微妙差异的 \Z 不同,\z 在其他多种正则表达式引擎中具有明确无歧义的解析方式。(由 Serhiy Storchaka 在 gh-133306 中贡献。)

  • 正则表达式 中,\B 现在可以匹配空输入字符串,其行为现在始终与 \b 相反。(由 Serhiy Storchaka 在 gh-124130 中贡献。)

  • iOS 和 macOS 应用程序现在可配置为将 stdoutstderr 输出内容重定向至系统日志。(由 Russell Keith-Magee 在 gh-127592 中贡献。)

  • iOS 测试平台现已支持在测试运行时实时流式传输测试输出,并可用于运行除 CPython 之外的其他项目测试套件。(由 Russell Keith-Magee 在 gh-127592 中贡献。)

  • 三参数 pow() 现在会在必要时尝试调用 __rpow__() 方法。此前该方法仅在双参数 pow() 和二元幂运算符中被调用。(由 Serhiy Storchaka 在 gh-130104 中贡献。)

  • 新增基于 HACL* 项目形式化验证代码的 HMAC (RFC 2104) 内置实现。当 OpenSSL 的 HMAC 实现不可用时,该实现将作为备用方案。(由 Bénédikt Tran 在 gh-99108 中贡献。)

  • 导入时间分析标志现可通过新增的 -X importtime=2 选项追踪已加载('缓存')模块。当导入此类模块时,selfcumulative 时间值将被替换为字符串 cached。当前规定 -X importtime 参数值大于 2 的选项保留供未来使用。(由 Noah Kim 和 Adam Turner 在 gh-118655 中贡献。)

  • 当从纯C类型派生子类时,若子类未显式重写相关方法,新建类型的C槽位将不再被替换为封装版本。(由 Tomasz Pytel 在 gh-132329 中贡献。)

  • 命令行选项 -c 现在会在执行前自动对其代码参数进行去缩进处理,该行为与 textwrap.dedent() 函数保持一致。(由 Jon Crall 和 Steven Sun 在 gh-103998 中贡献。)

  • 当使用 async with 而非 with 进入支持同步上下文管理器协议的对象时,错误提示信息已改进。反之,对于异步上下文管理器协议的情况也做了相应优化。(由 Bénédikt Tran 在 gh-128398 中贡献。)

  • -J 不再是 Jython 的保留标志,目前不具任何特殊含义。(由 Adam Turner 在 gh-133336 中贡献。)

  • 内置的 int() 不再委托给 __trunc__()。 希望支持转换为 int() 的类必须实现 __int__()__index__()。 (由 Mark Dickinson 在 gh-119743 中贡献。)

  • 在布尔上下文中使用 NotImplemented 现在会引发 TypeError 异常。该用法自 Python 3.9 起已引发 DeprecationWarning 弃用警告。(由 Jelle Zijlstra 在 gh-118767 中贡献。)

PEP 765:禁止在 finally 代码块中使用会跳出该块的 return/break/continue 语句

编译器在检测到 returnbreakcontinue 语句跳出 finally 代码块时,将触发 SyntaxWarning。此项变更遵循 PEP 765 规范。

新增模块

改进的模块

argparse

  • argparse.ArgumentParser程序名称 默认值现在会反映 Python 解释器定位 __main__ 模块代码的方式。(由 Serhiy Storchaka 和 Alyssa Coghlan 在 gh-66436 中贡献。)

  • argparse.ArgumentParser 新增可选形参 suggest_on_error,可在用户输入错误时提供参数选项及子解析器名称的建议。(由 Savannah Ostrowski 在 gh-124456 中贡献。)

  • 为帮助文本启用颜色,可以通过向 argparse.ArgumentParser 传递可选的 color 形参来禁用。这也可以通过 环境变量 来控制。(由 Hugo van Kemenade 在 gh-130645 中贡献。)

ast(抽象语法树)

  • 新增 ast.compare() 函数用于比较两个抽象语法树(AST)。(由 Batuhan Taskaya 和 Jeremy Hylton 在 gh-60191 中贡献。)

  • 新增对AST节点 copy.replace() 操作的支持。(由 Bénédikt Tran 在 gh-121141 中贡献。)

  • 在优化级别为2时,文档字符串现会从优化后的AST中移除。(由 Irit Katriel 在 gh-123958 中贡献。)

  • AST节点的 repr() 输出现在包含更多信息。(由 Tomas Roun 在 gh-116022 中贡献。)

  • 当传入AST作为输入时,ast.parse() 现在会始终验证根节点类型是否合规。(由 Irit Katriel 在 gh-130139 中贡献。)

  • 新增命令行接口选项 --feature-version--optimize--show-empty。(由 Semyon Moroz 在 gh-133367 中贡献。)

asyncio

calendar(日历)

  • 默认情况下,calendar 模块的 命令行 文本输出会以彩色高亮显示当日日期,该功能可通过 环境变量 控制。(由 Hugo van Kemenade 在 gh-128317 中贡献。)

concurrent.futures

  • 新增 InterpreterPoolExecutor 类,该功能向Python代码暴露"子解释器"能力(同一进程中的多个Python解释器)。此实现与 PEP 734 提案的API相互独立。(由 Eric Snow 在 gh-124548 中贡献。)

configparser

contextvars

ctypes

curses

datetime

decimal

difflib

  • difflib.HtmlDiff 类生成的高亮显示更改的比较页面现在支持暗色模式。 (由李佳昊在 gh-129939 中贡献。)

dis

errno

faulthandler

fnmatch

fractions

functools

getopt

  • 新增对可选参数选项的支持。(由 Serhiy Storchaka 在 gh-126374 中贡献。)

  • 新增对按顺序返回混合选项和非选项参数的支持。(由 Serhiy Storchaka 在 gh-126390 中贡献。)

getpass

  • 新增 getpass.getpass() 通过仅限关键字可选参数 echo_char 提供键盘反馈的功能。每当输入字符时会显示占位符,删除字符时则会移除该占位符。(由 Semyon Moroz 在 gh-77065 中贡献。)

graphlib

heapq

hmac

  • 新增基于 HACL* 项目形式化验证代码的 HMAC (RFC 2104) 内置实现。(由 Bénédikt Tran 在 gh-99108 中贡献。)

http

  • http.server 模块生成的目录列表和错误页面允许浏览器应用其默认暗模式。(由 Yorik Hansen 在 gh-123430 中贡献。)

  • http.server 模块现在支持使用 http.server.HTTPSServer 类通过 HTTPS 提供服务。此功能通过命令行界面 (python -m http.server) 通过以下选项公开:

    • --tls-cert <path>: TLS 证书文件的路径。

    • --tls-key <path>: 私钥文件的可选路径。

    • --tls-password-file <path>: 私钥密码文件的可选路径。

    (由 Semyon Moroz 在 gh-85162 中贡献。)

imaplib

inspect

io

json

  • 添加关于 JSON 序列化错误的说明,以便识别错误来源。(由 Serhiy Storchaka 在 gh-122163 中贡献。)

  • 使 json 模块能够使用 -m 开关作为脚本运行:python -m json。请参阅 JSON 命令行界面 文档。(由 Trey Hunner 在 gh-122873 中贡献。)

  • 默认情况下,JSON 命令行界面 的输出会以彩色突出显示。这可以通过 环境变量 进行控制。(由 Tomas Roun 在 gh-131952 中贡献。)

linecache

logging.handlers

math

  • 为模块中的域错误添加了更详细的错误消息。(由 Charlie Zhao 和 Sergey B Kirpichev 在 gh-101410 中贡献。)

mimetypes

  • 完善 mimetypes 模块的命令行文档。现在,在失败时返回码由 0 改为 1,在命令行形参错误时返回码由 1 改为 2。此外,错误会输出到 stderr 而非 stdout,且错误文本更为简洁。(由 Oleg Iarygin 和 Hugo van Kemenade 在 gh-93096 中贡献。)

  • 添加用于字体的 MS 和 RFC 8081 MIME 类型:

    • 嵌入式 OpenType: application/vnd.ms-fontobject

    • OpenType 布局(OTF) font/otf

    • TrueType: font/ttf

    • WOFF 1.0 font/woff

    • WOFF 2.0 font/woff2

    (由 Sahil Prajapati 和 Hugo van Kemenade 在 gh-84852 中贡献。)

  • 添加用于 Matroska 视听数据容器结构的 RFC 9559 MIME 类型,包括:

    • 仅有音频,无视频: audio/matroska (.mka)

    • 视频:video/matroska (.mkv)

    • 立体视频: video/matroska-3d (.mk3d)

    (由 Hugo van Kemenade 在 gh-89416 中贡献。)

  • 新增符合RFC标准的图像MIME类型:

    • RFC 1494: CCITT Group 3 (.g3)

    • RFC 3362: Real-time Facsimile, T.38 (.t38)

    • RFC 3745: JPEG 2000 (.jp2)、扩展格式 (.jpx) 和复合格式 (.jpm)

    • RFC 3950: 标签图像文件格式传真扩展,TIFF-FX (.tfx)

    • RFC 4047: 灵活图像传输系统 (.fits)

    • RFC 7903: 增强型图元文件 (.emf) 和 Windows 图元文件 (.wmf)

    (由 Hugo van Kemenade 在 gh-85957 中贡献。)

  • 更多 MIME 类型更改:

    • RFC 2361: 将 .avi 的类型更改为 video/vnd.avi,将 .wav 的类型更改为 audio/vnd.wave

    • RFC 4337:添加 MPEG-4 audio/mp4 (.m4a)

    • RFC 5334:添加 Ogg 媒体(.oga.ogg.ogx

    • RFC 6713:添加 application/gzip (.gz)

    • RFC 9639: 添加 FLAC 格式的 audio/flac (.flac)

    • 添加 7z application/x-7z-compressed (.7z)

    • 非严格模式下,添加Android安装包 application/vnd.android.package-archive (.apk)

    • 添加 deb application/x-debian-package (.deb)

    • 添加 glTF 二进制格式 model/gltf-binary (.glb)

    • 添加glTF JSON/ASCII格式 model/gltf+json (.gltf)

    • 添加 M4V video/x-m4v (.m4v)

    • 添加 PHP application/x-httpd-php (.php)

    • 添加 RAR application/vnd.rar (.rar)

    • 添加 RPM application/x-rpm (.rpm)

    • 添加STL格式 model/stl (.stl)

    • 添加 Windows 媒体视频 video/x-ms-wmv (.wmv)

    • 事实上:添加WebM格式 audio/webm (.weba)

    • ECMA-376: 添加 .docx.pptx.xlsx 类型

    • OASIS: 添加 OpenDocument 格式的 .odg.odp.ods.odt 类型

    • W3C: 添加 EPUB 格式的 application/epub+zip (.epub)

    (由 Hugo van Kemenade 在 gh-129965 中贡献。)

  • 添加 RFC 9512 中针对 YAML 文件(.yaml.yml)的 application/yaml MIME 类型。(由 Sasha "Nelie" Chernykh 和 Hugo van Kemenade 在 gh-132056 中贡献。)

multiprocessing

  • 默认的 启动方法 在 macOS 和 Windows 以外的平台上已从 fork 改为 forkserver,而在 macOS 和 Windows 平台上,该方法原本就是 spawn

    如果需要使用与线程不兼容的 fork 方法,则必须通过 multiprocessing.get_context() 提供的上下文显式请求该方法(推荐方式),或者通过 multiprocessing.set_start_method() 更改默认方法。

    请参阅 forkserver 限制说明 了解与 fork 方法的差异信息,以及此项变更对存在以下情况的现有代码可能产生的影响:(1) 使用可变全局共享变量 (2) 包含无法被 pickle 自动序列化的共享对象。

    (由 Gregory P. Smith 在 gh-84559 中贡献)

  • multiprocessing"forkserver" 启动方法现增加了控制套接字认证机制,不再仅依赖文件系统权限来限制其他进程可能触发的衍生工作进程及运行代码。(由 Gregory P. Smith 在 gh-97514 中贡献。)

  • 针对 listdict 类型的 多进程代理对象 增加了此前被忽略的缺失方法:

    • clear()copy() 用于 list 的代理

    • fromkeys()reversed(d)d | {}{} | dd |= {'b': 2} 用于 dict 的代理

    (由 Roy Hyunjin Han 在 gh-103134 中贡献。)

  • 通过 SyncManager.set() 增加了对共享 set 对象的支持。multiprocessing.Manager() 方法中的 set() 现已可用。(由 Mingyu Park 在 gh-129949 中贡献。)

  • 新增 multiprocessing.Process.interrupt() 方法,该方法通过发送 SIGINT 信号来终止子进程。这使得 finally 子句能够为被终止的进程打印栈回溯信息。(由 Artem Pulkin 在 gh-131913 中贡献。)

operator

  • 新增了两个函数: operator.is_none()operator.is_not_none(),其中 operator.is_none(obj) 等价于 obj is None,而 operator.is_not_none(obj) 等价于 obj is not None。(由 Raymond Hettinger 和 Nico Mexis 在 gh-115808 中贡献。)

os

os.path

pathlib

  • pathlib.Path 新增递归复制/移动文件和目录的方法:

    • copy() 将一个文件或目录树复制到目标位置。

    • copy_into() 会将内容复制 目标目录中。

    • move() 将一个文件或目录树移动到目标位置。

    • move_into() 会将内容移动 目标目录中。

    (由 Barney Gale 在 gh-73991 中贡献。)

  • 新增 pathlib.Path.info 属性,该属性存储一个实现了 pathlib.types.PathInfo 协议(同样为新增)的对象。此对象支持查询文件类型,并在内部缓存 stat() 的结果。通过 iterdir() 生成的路径对象会利用扫描父目录时收集到的文件类型信息进行初始化。(由 Barney Gale 在 gh-125413 中贡献。)

pdb

  • 硬编码断点(breakpoint()pdb.set_trace())现在会重用最近一次调用 set_trace()Pdb 实例,而不是每次都创建一个新实例。因此,所有实例特定的数据(如 displaycommands)在硬编码断点之间都会被保留下来。(由 Tian Gao 在 gh-121450 中贡献。)

  • pdb.Pdb 新增一个 mode 参数。当 pdb 处于 inline 模式时,禁用 restart 命令。(由 Tian Gao 在 gh-123757 中贡献。)

  • 当用户尝试在 inline 模式下退出 pdb 时,会显示一个确认提示。输入 yY、按 <Enter>EOF 将确认退出并调用 sys.exit(),而非引发 bdb.BdbQuit。(由 Tian Gao 在 gh-124704 中贡献。)

  • breakpoint()pdb.set_trace() 这样的内联断点将始终在调用帧处暂停程序,而忽略 skip 模式(如果有的话)。(由 Tian Gao 在 gh-130493 中贡献。)

  • pdb 的多行输入中,行首的 <tab> 键现在会填充 4 个空格的缩进,而不是插入 \t 字符。(由 Tian Gao 在 gh-130471 中贡献。)

  • pdb 的多行输入中引入了自动缩进功能。当检测到新的代码块时,它会要么保持上一行的缩进,要么插入 4 个空格的缩进。(由 Tian Gao 在 gh-133350 中贡献。)

  • 新增 $_asynctask 以在适用情况下访问当前的 asyncio 任务。(由 Tian Gao 在 gh-124367 中贡献。)

  • 新增 pdb.set_trace_async() 以支持调试 asyncio 协程。此函数支持 await 语句。(由 Tian Gao 在 gh-132576 中贡献。)

  • pdb 中显示的源代码将带有语法高亮。除了新添加的 pdb.Pdbcolorize 参数外,还可以使用与 PyREPL 相同的方法来控制此特性。(由 Tian Gao 和 Łukasz Langa 在 gh-133355 中贡献。)

pickle

  • pickle 模块的默认协议版本设置为 5。更多详细信息,请参见 pickle protocols

  • 添加了关于 pickle 序列化错误的说明,这些说明有助于识别错误的来源。(由 Serhiy Storchaka 在 gh-122213 中贡献。)

platform

pydoc

  • 帮助输出中的 注解 现在通常以更接近原始源代码中的格式显示。(由 Jelle Zijlstra 在 gh-101552 中贡献。)

socket

  • 改进并修复对蓝牙套接字的支持。

    • 修复了 NetBSD 和 DragonFly BSD 系统上对蓝牙套接字的支持。(由 Serhiy Storchaka 在 gh-132429 中贡献。)

    • 修复了 FreeBSD 系统上对 BTPROTO_HCI 的支持。(由 Victor Stinner 在 gh-111178 中贡献。)

    • 新增对 FreeBSD 系统上 BTPROTO_SCO 的支持。(由 Serhiy Storchaka 在 gh-85302 中贡献。)

    • 新增对 FreeBSD 系统上 BTPROTO_L2CAP 地址中 cidbdaddr_type 的支持。(由 Serhiy Storchaka 在 gh-132429 中贡献。)

    • 新增对 Linux 系统上 BTPROTO_HCI 地址中 channel 的支持。(由 Serhiy Storchaka 在 gh-70145 中贡献。)

    • 在 Linux 系统上,允许将整数作为 BTPROTO_HCI 的地址。(由 Serhiy Storchaka 在 gh-132099 中贡献。)

    • BTPROTO_L2CAP 中,getsockname() 会返回 cid。(由 Serhiy Storchaka 在 gh-132429 中贡献。)

    • 新增了许多常量。(由 Serhiy Storchaka 在 gh-132734 中贡献。)

ssl

  • 通过 ssl.HAS_PHA 指示 ssl 模块是否支持 TLSv1.3 握手后客户端认证(PHA)。(由 Will Childs-Klein 在 gh-128036 中贡献。)

struct

  • struct 模块中支持 float complexdouble complex 这两种 C 类型(分别对应格式字符 'F''D')。(由 Sergey B Kirpichev 在 gh-121249 中贡献。)

symtable

sys

  • 之前未写入文档的特殊函数 sys.getobjects(),它仅存在于某些专用的 Python 构建版,现在可以从其他解释器而非调用它的解释器返回对象。

  • 新增 sys._is_immortal() 函数,用于判断一个对象是否为 immortal。 (由 Peter Bierma 在 gh-128509 中贡献。)

  • 在 FreeBSD 上,sys.platform 将不再包含主版本号。 它将始终为 'freebsd',而不是 'freebsd13''freebsd14'

  • sys._clear_type_cache() 函数引发 DeprecationWarning 警告。该函数在 Python 3.13 中已被弃用,但此前并未引发运行时警告。

sys.monitoring

sysconfig

tarfile

threading

tkinter

  • 使 tkinter 控件的 after()after_idle() 方法支持通过关键字传递参数。(由 Zhikang Yan 在 gh-126899 中贡献。)

  • 新增为 tkinter.OptionMenutkinter.ttk.OptionMenu 指定名称的功能。(由 Zhikang Yan 在 gh-130482 中贡献。)

turtle(海龟绘图)

types(类型)

typing

  • types.UnionTypetyping.Union 现在互为别名,这意味着旧式联合(通过 Union[int, str] 创建)和新式联合(int | str)现在都会创建相同运行时类型的实例。这统一了两种语法的行为,但也带来了一些行为差异,可能会影响那些在运行时对类型进行内省的用户:

    • 创建联合的两种语法现在在 repr() 中会生成相同的字符串表示形式。例如,repr(Union[int, str]) 现在会返回 "int | str",而非 "typing.Union[int, str]"

    • 使用旧式语法创建的联合不再被缓存。此前,多次运行 Union[int, str] 会返回同一个对象(Union[int, str] is Union[int, str] 的结果为 True),但现在会返回两个不同的对象。用户应当使用 == 来比较联合是否相等,而非 is。新式联合从未以这种方式被缓存。 这一变化可能会增加某些程序的内存占用,这些程序会通过下标 typing.Union 创建大量联合。不过,有几个因素会抵消这一开销:由于 PEP 649 的引入,Python 3.14 中默认不再对注解中使用的联合进行求值;types.UnionType 的实例本身比早期 Python 版本中 Union[] 返回的对象小得多;此外,移除缓存也节省了部分空间。因此,对于大多数用户而言,这一变化不太可能导致内存占用显著增加。

    • 此前,旧式联合是通过私有类 typing._UnionGenericAlias 实现的。该类在当前实现中已不再需要,但为了向后兼容性而被保留,计划在 Python 3.17 中移除。用户应使用有文档记录的内省辅助函数,如 typing.get_origin()typing.get_args(),而非依赖私有实现细节。

    • 现在可以在 isinstance() 检查中使用 typing.Union 本身。例如,isinstance(int | str, typing.Union) 将返回 True;而此前这会引发 TypeError

    • typing.Union 对象的 __args__ 属性不再可写。

    • typing.Union 对象现在禁止设置任何属性。此前版本中仅能设置双下划线属性,且该行为既无文档支持,实际使用中也存在诸多潜在问题。

    (由 Jelle Zijlstra 在 gh-105499 中贡献。)

  • typing.TypeAliasType 现在支持星号解包操作。

unicodedata

  • Unicode 数据库已更新到 16.0.0 版本。

unittest

urllib

  • 升级 urllib.request 的 HTTP 摘要认证算法,支持 RFC 7616 中规定的 SHA-256 摘要认证。(由 Calvin Bui 在 gh-128193 中贡献。)

  • 改进解析和生成 file: URL 时的易用性和标准合规性。

    urllib.request.url2pathname()

    • 当新的 require_scheme 参数设为 true 时,接受完整的 URL。

    • 如果 URL 的权限部分与本地主机名匹配,则舍弃该部分。

    • 当新的 resolve_host 参数设为 true 时,若 URL 的权限部分解析为本地 IP 地址,则舍弃该部分。

    • 丢弃URL查询和片段组件。

    • 如果 URL 的权限部分不是本地的,则引发 URLError,但在 Windows 系统上,仍会像以前一样返回 UNC 路径。

    urllib.request.pathname2url() 中:

    • 当新的 add_scheme 参数设为 true 时,返回完整的 URL。

    • 当路径以斜杠开头时,包含一个空的 URL 权限部分。例如,路径 /etc/hosts 会被转换为 URL ///etc/hosts

    在 Windows 系统上,盘符不再转换为大写,且非紧跟在驱动器号后的 : 字符不再引发 OSError 异常。

    (由 Barney Gale 在 gh-125866 中贡献。)

uuid

webbrowser

  • BROWSER 环境变量中的名称现在可以引用 webbrowser 模块中已注册的浏览器,而不必总是生成新的浏览器命令。

    这使得可以将 BROWSER 设置为 macOS 上受支持的浏览器之一的值。

zipfile

性能优化

asyncio

  • 标准基准测试结果提升了10-20%,这是在为 native tasks 实现新的每线程双链表后取得的,同时也减少了内存使用。这使得诸如 python -m asyncio pstree 等外部内省工具能够内省在所有线程中运行的 asyncio 任务的调用图。(由 Kumar Aditya 在 gh-107803 中贡献。)

  • 该模块现在对 自由线程构建 提供了一流的支持。这使得多个事件循环可以在不同线程中并行执行,并且随着线程数量的增加线性扩展。(由 Kumar Aditya 在 gh-128002 中贡献。)

base64

  • b16decode() 的速度现在提升了高达六倍。(由 Bénédikt Tran、Chris Markiewicz 和 Adam Turner 在 gh-118761 中贡献。)

bdb

  • 基本调试器现在有了基于 sys.monitoring 的后端,可以通过将 'monitoring' 传递给 Bdb 类的新 backend 参数来选择。(由 Tian Gao 在 gh-124533 中贡献。)

difflib

  • IS_LINE_JUNK() 函数的速度现在提升了一倍。(由 Adam Turner 和 Semyon Moroz 在 gh-130167 中贡献。)

gc

  • 新的 增量垃圾回收器 意味着对于较大的堆,最大暂停时间减少了至少一个数量级。

    由于这种优化,get_threshold()set_threshold() 的结果含义发生了变化,同时还有 get_count()get_stats()

    • 为了向后兼容,get_threshold() 继续返回一个三项元组。第一个值是年轻代回收的阈值,与之前相同;第二个值决定了老年代回收的扫描速率(默认值为10,值越高意味着老年代回收扫描越慢)。第三个值现在没有意义,总是为零。

    • set_threshold() 现在会忽略第二个之后的任何项。

    • get_count()get_stats() 继续返回相同格式的结果。唯一的区别是,结果不再指向年轻代、老化代和老年代,而是指向年轻代以及老年代的老化和回收空间。

    总的来说,试图操纵循环垃圾回收器行为的代码可能不会完全按预期工作,但极不可能造成危害。所有其他代码将正常工作。

    (由 Mark Shannon 在 gh-108362 中贡献。)

io

  • 打开和读取文件现在执行更少的系统调用。完整读取一个小型操作系统缓存文件的速度提高了最多15%。(由 Cody Maloney 和 Victor Stinner 在 gh-120754gh-90102 中贡献。)

pathlib

  • Path.read_bytes 现在使用无缓冲模式打开文件,完整读取的速度提高了 9% 到 17%。(由 Cody Maloney 在 gh-120754 中贡献。)

pdb

uuid

  • uuid3()uuid5() 对于16字节名称的速度现在大约提高了40%,对于1024字节名称的速度提高了20%。更长名称的性能保持不变。(由 Bénédikt Tran 在 gh-128150 中贡献。)

  • uuid4() 的速度现在提高了约30%。(由 Bénédikt Tran 在 gh-128150 中贡献。)

zlib

  • 在 Windows 上,zlib-ng 现在用作默认二进制文件中 zlib 模块的实现。已知 zlib-ng 与之前使用的 zlib 实现之间没有不兼容性。这应在所有压缩级别上带来更好的性能。

    值得注意的是,zlib.Z_BEST_SPEED1) 可能会导致比之前的实现显著更低的压缩率,同时显著减少压缩所需的时间。

    (由 Steve Dower 在 gh-91349 中贡献。)

移除

argparse

  • 移除 BooleanOptionalAction`的 *type*, *choices* *metavar* 形参。 它们自 Python 3.12 起已被弃用。 (由 Nikita Sobolev :gh:`118805 中贡献。)

  • 在参数组上调用 add_argument_group() 现在会引发 ValueError。 类似地,在互斥组上调用 add_argument_group()add_mutually_exclusive_group() 现在都会引发 ValueError。 这种‘嵌套调用’从未被支持,经常无法正常工作,且通过继承关系被意外暴露。 该功能自 Python 3.11 起已被弃用。 (由 Savannah Ostrowski 在 gh-127186 中贡献。)

ast(抽象语法树)

  • 移除以下自 Python 3.8 起作为 Constant 的已弃用别名,且自 Python 3.12 起已发出弃用警告的类:

    • Bytes

    • Ellipsis

    • NameConstant

    • Num

    • Str

    由于这些移除操作,当自定义的 NodeVisitor 子类访问抽象语法树(AST)时,用户定义的 visit_Numvisit_Strvisit_Bytesvisit_NameConstantvisit_Ellipsis 方法将不再被调用。请改用定义 visit_Constant 方法。

    (由 Alex Waygood 在 gh-119562 中贡献。)

  • 移除以下 ast.Constant 为兼容现已移除的 AST 类而保留的已弃用属性:

    • Constant.n

    • Constant.s

    改用 Constant.value。 (由 Alex Waygood 在 gh-119562 中贡献。)

asyncio

  • 移除以下自 Python 3.12 起已被弃用的类、方法和函数:

    • AbstractChildWatcher

    • FastChildWatcher

    • MultiLoopChildWatcher

    • PidfdChildWatcher

    • SafeChildWatcher

    • ThreadedChildWatcher

    • AbstractEventLoopPolicy.get_child_watcher()

    • AbstractEventLoopPolicy.set_child_watcher()

    • get_child_watcher()

    • set_child_watcher()

    (由 Kumar Aditya 在 gh-120804 中贡献。)

  • asyncio.get_event_loop() 现在如果没有当前事件循环,会引发 RuntimeError 异常,且不再隐式创建事件循环。

    (由 Kumar Aditya 在 gh-126353 中贡献。)

    当前存在几种使用 asyncio.get_event_loop() 的模式,其中大多数可替换为 asyncio.run()

    如果正在运行异步函数,直接使用 asyncio.run() 即可。

    之前:

    async def main():
        ...
    
    
    loop = asyncio.get_event_loop()
    try:
        loop.run_until_complete(main())
    finally:
        loop.close()
    

    之后:

    async def main():
        ...
    
    asyncio.run(main())
    

    如果需要启动某些持续运行的服务,例如监听套接字的服务器,请使用 asyncio.run() 配合 asyncio.Event 实现。

    之前:

    def start_server(loop): ...
    
    loop = asyncio.get_event_loop()
    try:
        start_server(loop)
        loop.run_forever()
    finally:
        loop.close()
    

    之后:

    def start_server(loop): ...
    
    async def main():
        start_server(asyncio.get_running_loop())
        await asyncio.Event().wait()
    
    asyncio.run(main())
    

    如果需要在事件循环中运行某些任务,同时在其前后执行阻塞代码,请使用 asyncio.Runner

    之前:

    async def operation_one(): ...
    def blocking_code(): ...
    async def operation_two(): ...
    
    loop = asyncio.get_event_loop()
    try:
        loop.run_until_complete(operation_one())
        blocking_code()
        loop.run_until_complete(operation_two())
    finally:
        loop.close()
    

    之后:

    async def operation_one(): ...
    def blocking_code(): ...
    async def operation_two(): ...
    
    with asyncio.Runner() as runner:
        runner.run(operation_one())
        blocking_code()
        runner.run(operation_two())
    

collections.abc

  • 移除 ByteString,它自 Python 3.12 起已被弃用。 (由 Nikita Sobolev 在 gh-118803 中贡献。)

email

  • 移除 email.utils.localtime() 函数的 isdst 形参。该参数自 Python 3.12 起已被弃用且一直被忽略。(由 Hugo van Kemenade 在 gh-118798 中贡献。)

importlib.abc

itertools

  • 移除 itertools 迭代器对复制(copy)、深度复制(deepcopy)和序列化(pickle)操作的支持。自 Python 3.12 起,这些操作已触发 DeprecationWarning 警告。(由 Raymond Hettinger 在 gh-101588 中贡献。)

pathlib

  • 移除向 Path 传递额外关键字参数的支持。在先前版本中,此类参数均会被忽略。(由 Barney Gale 在 gh-74033 中贡献。)

  • 移除向 PurePath.relative_to()is_relative_to() 方法传递额外位置参数的支持。在先前版本中,此类参数会被拼接到 other 参数上。(由 Barney Gale 在 gh-78707 中贡献。)

pkgutil

  • 移除 get_loader()find_loader() 函数,这两个函数自 Python 3.12 起已被弃用。(由 Bénédikt Tran 在 gh-97850 中贡献。)

pty

  • 移除 master_open()slave_open() 函数,这两个函数自 Python 3.12 起已被弃用。请改用 pty.openpty() 函数。

sqlite3

typing

  • 移除 ByteString,它自 Python 3.12 起已被弃用。 (由 Nikita Sobolev 在 gh-118803 中贡献。)

urllib

  • urllib.parse 模块中移除 Quoter 类,该类自 Python 3.11 起已被弃用。(由 Nikita Sobolev 在 gh-118827 中贡献。)

  • urllib.request 模块中移除 URLopenerFancyURLopener 类,这两个类自 Python 3.3 起已被弃用。

    myopener.open() 可以替换为 urlopen()myopener.retrieve() 可以替换为 urlretrieve()。对 opener 类的自定义操作可以通过向 build_opener() 传递定制化的处理程序来实现替换。(由 Barney Gale 在 gh-84850 中贡献。)

弃用

计划在 Python 3.15 中移除

  • 导入系统:

    • 当设置 __spec__.cached 失败时在模块上设置 __cached__ 的做法已被弃用。 在 Python 3.15 中,__cached__ 将不会再被导入系统或标准库纳入考虑。 (gh-97879)

    • 当设备 __spec__.parent 失败时在模块上设置 __package__ 的做法已被弃用。 在 Python 3.15 中,__package__ 将不会再被导入系统或标准库纳入考虑。 (gh-97879)

  • ctypes:

    • 未写入文档的 ctypes.SetPointerType() 函数自 Python 3.13 起已被弃用。

  • http.server:

    • The obsolete and rarely used CGIHTTPRequestHandler has been deprecated since Python 3.13. No direct replacement exists. Anything is better than CGI to interface a web server with a request handler.

    • 用于 python -m http.server 命令行界面的 --cgi 旗标自 Python 3.13 起已被弃用。

  • importlib:

    • load_module() 方法:改用 exec_module()

  • locale:

  • pathlib:

    • .PurePath.is_reserved() has been deprecated since Python 3.13. Use os.path.isreserved() to detect reserved paths on Windows.

  • platform:

    • platform.java_ver() has been deprecated since Python 3.13. This function is only useful for Jython support, has a confusing API, and is largely untested.

  • sysconfig:

  • threading:

    • 在 Python 3.15 中 RLock() 将不再接受参数。 传入参数的做法自 Python 3.14 起已被弃用,因为 Python 版本不接受任何参数,而 C 版本允许任意数量的位置或关键字参数,但会忽略所有参数。

  • types:

  • typing:

    • 未写入文档的用于创建 NamedTuple 类的关键字参数语法(例如 Point = NamedTuple("Point", x=int, y=int))自 Python 3.13 起已被弃用。 请改用基于类的语法或函数语法。

    • 当使用 TypedDict 的函数式语法时,不向 fields 形参传递值 (TD = TypedDict("TD")) 或传递 None (TD = TypedDict("TD", None)) 的做法自 Python 3.13 起已被弃用。 请改用 class TD(TypedDict): passTD = TypedDict("TD", {}) 来创建一个零字段的 TypedDict。

    • typing.no_type_check_decorator() 装饰器自 Python 3.13 起已被弃用。 存在于 typing 模块八年之后,它仍未被任何主要类型检查器所支持。

  • sre_compile, sre_constantssre_parse 模块。

  • wave

    • The getmark(), setmark() and getmarkers() methods of the Wave_read and Wave_write classes have been deprecated since Python 3.13.

  • zipimport:

    • zipimport.zipimporter.load_module() has been deprecated since Python 3.10. Use exec_module() instead. (gh-125746.)

计划在 Python 3.16 中移除

计划在 Python 3.17 中移除

  • typing:

    • 在 Python 3.14 之前,旧式的联合是通过私有类 typing._UnionGenericAlias 实现的。实现已不再需要该类,但为向后兼容性保留了该类,并计划在 Python 3.17 中删除。 用户应使用记录在案的内省助手函数,如 typing.get_origin()typing.get_args(),而不是依赖于私有的实现细节。

计划在 Python 3.19 中移除

  • ctypes:

    • 在非 Windows 平台上,通过设置 _pack_ 而非 _layout_ ,隐式切换到与 MSVC 兼容的结构布局。

  • hashlib:

    • In hash function constructors such as new() or the direct hash-named constructors such as md5() and sha256(), their optional initial data parameter could also be passed a keyword argument named data= or string= in various hashlib implementations.

      Support for the string keyword argument name is now deprecated and slated for removal in Python 3.19.

      Before Python 3.13, the string keyword parameter was not correctly supported depending on the backend implementation of hash functions. Prefer passing the initial data as a positional argument for maximum backwards compatibility.

计划在未来版本中移除

以下API将会被移除,尽管具体时间还未确定。

  • argparse

    • 嵌套参数组和嵌套互斥组已被弃用 。

    • 将未写入文档的关键字参数 prefix_chars 传递给 add_argument_group() 的做法现在已被弃用。

    • argparse.FileType 类型转换器已弃用 。

  • builtins:

    • bool(NotImplemented)

    • 生成器: throw(type, exc, tb)athrow(type, exc, tb) 签名已被弃用:请改用 throw(exc)athrow(exc),即单参数签名。

    • 目前 Python 接受数字类字面值后面紧跟关键字的写法,例如 0in x, 1or x, 0if 1else 2。 它允许像 [0x1for x in y] 这样令人困惑且有歧义的表达式 (它可以被解读为 [0x1 for x in y] 或者 [0x1f or x in y])。 如果数字类字面值后面紧跟关键字 and, else, for, if, in, isor 中的一个将会引发语法警告。 在未来的版本中它将改为语法错误。 (gh-87999)

    • __index__()__int__() 方法返回非 int 类型的支持:将要求这些方法必须返回 int 的子类的实例。

    • __float__() 方法返回 float 的子类的支持:将要求这些方法必须返回 float 的实例。

    • __complex__() 方法返回 complex 的子类的支持:将要求这些方法必须返回 complex 的实例。

    • int() 委托给 __trunc__() 方法。

    • 传入一个复数作为 complex() 构造器中的 realimag 参数的做法现在已被弃用;它应当仅作为单个位置参数被传入。 (由 Serhiy Storchaka 在 gh-109218 中贡献。).)

  • calendar: calendar.Januarycalendar.February 常量已被弃用并由 calendar.JANUARYcalendar.FEBRUARY 替代。 (由 Prince Roshan 在 gh-103636 中贡献。)

  • codecscodecs.open() 请改用 open() 。 (gh-133038)

  • codeobject.co_lnotab: 改用 codeobject.co_lines() 方法。

  • datetime:

    • utcnow(): 使用 datetime.datetime.now(tz=datetime.UTC)

    • utcfromtimestamp(): 使用 datetime.datetime.fromtimestamp(timestamp, tz=datetime.UTC)

  • gettext: 复数值必须是一个整数。

  • importlib:

  • importlib.metadata:

    • EntryPoints 元组接口。

    • 返回值中隐式的 None

  • logging: warn() 方法自 Python 3.3 起已被弃用,请改用 warning()

  • mailbox: 对 StringIO 输入和文本模式的使用已被弃用,改用 BytesIO 和二进制模式。

  • os: 在多线程的进程中调用 os.register_at_fork()

  • pydoc.ErrorDuringImport: 使用元组值作为 exc_info 形参的做法已被弃用,应使用异常实例。

  • re: 现在对于正则表达式中的数字分组引用和分组名称将应用更严格的规则。 现在只接受 ASCII 数字序列作为数字引用。 字节串模式和替换字符串中的分组名称现在只能包含 ASCII 字母和数字以及下划线。 (由 Serhiy Storchaka 在 gh-91760 中贡献。)

  • shutil: rmtree()onerror 形参在 Python 3.12 中已被弃用;请改用 onexc 形参。

  • ssl 选项和协议:

    • ssl.SSLContext 不带 protocol 参数的做法已被弃用。

    • ssl.SSLContext: set_npn_protocols()selected_npn_protocol() 已被弃用:请改用 ALPN。

    • ssl.OP_NO_SSL* 选项

    • ssl.OP_NO_TLS* 选项

    • ssl.PROTOCOL_SSLv3

    • ssl.PROTOCOL_TLS

    • ssl.PROTOCOL_TLSv1

    • ssl.PROTOCOL_TLSv1_1

    • ssl.PROTOCOL_TLSv1_2

    • ssl.TLSVersion.SSLv3

    • ssl.TLSVersion.TLSv1

    • ssl.TLSVersion.TLSv1_1

  • threading 的方法:

  • typing.Text (gh-92332)。

  • 内部类 typing._UnionGenericAlias 不再用于实现 typing.Union。为了保护使用该私有类的用户的兼容性 ,将至少在 Python 3.17 之前提供兼容性。 (由 Jelle Zijlstra 在 gh-105499 中贡献。)

  • unittest.IsolatedAsyncioTestCase: 从测试用例返回不为 None 的值的做法已被弃用。

  • urllib.parse 函数已被弃用:改用 urlparse()

    • splitattr()

    • splithost()

    • splitnport()

    • splitpasswd()

    • splitport()

    • splitquery()

    • splittag()

    • splittype()

    • splituser()

    • splitvalue()

    • to_bytes()

  • wsgiref: SimpleHandler.stdout.write() 不应执行部分写入。

  • xml.etree.ElementTree: 对 Element 的真值测试已被弃用。 在未来的发布版中它将始终返回 True。 建议改用显式的 len(elem)elem is not None 测试。

  • sys._clear_type_cache() 已弃用,请改用 sys._clear_internal_caches()

CPython 字节码的改变

  • 将操作码 BINARY_SUBSCR 替换为带有操作数 NB_SUBSCRBINARY_OP 操作码。(由 Irit Katriel 在 gh-100239 中贡献。)

构建变化

  • 生成 configure 文件现在需要 GNU Autoconf 2.72 版本。(由 Erlend Aasland 在 gh-115765 中贡献。)

  • 现在可以通过 Py_NO_LINK_LIB 关闭基于 #pragmapython3*.lib 的链接。(由 Jean-Christophe Fillion-Robin 在 gh-82909 中贡献。)

PEP 761:停止使用 PGP 签名

从 CPython 3.14 版本开始,将不再提供 PGP 签名。验证发布产物的用户必须使用 Sigstore 验证材料 来验证 CPython 的发布包。这一发布流程的变更已在 PEP 761 中明确规定。

C API 的变化

新的特性

受限 C API 的变化

弃用

计划在 Python 3.15 中移除

计划在 Python 3.16 中移除

  • 捆绑的 libmpdec 副本。

计划在 Python 3.18 中移除

计划在未来版本中移除

以下 API 已被弃用,将被移除,但目前尚未确定移除日期。

移除

  • 使用可变基类创建 不可变类型 的行为自 Python 3.12 起已被弃用,现会引发 TypeError

  • 移除自 Python 3.12 起已被弃用的 PyDictObject.ma_version_tag 成员变量,请改用 PyDict_AddWatcher() API。 (由 Sam Gross 在 gh-124296 中贡献。)

  • 移除私有函数 _Py_InitializeMain()。该函数是 PEP 587 在 Python 3.8 中引入的 provisional API。 (由 Victor Stinner 在 gh-129033 中贡献。)

  • 移除 Python 3.13 中新增的未文档化 API: Py_C_RECURSION_LIMITPyThreadState.c_recursion_remaining,未经弃用期直接移除。 请在 C 代码中使用 Py_EnterRecursiveCall() 来防范无限递归问题。(移除于 gh-133079,另请参见 gh-130396。)

移植到 Python 3.14

本节列出了先前描述的更改以及可能需要更改代码的其他错误修正.

Python API 的变化

  • functools.partial 现在是一个方法描述符。若需保持原有行为,请将其包装在 staticmethod() 中。(由 Serhiy Storchaka 和 Dominykas Grigonis 在 gh-121027 中贡献。)

  • 垃圾回收器现在是增量的,这意味着 gc.collect() 的行为略有变化:

    • gc.collect(1): 执行增量垃圾回收,而不是回收第 1 代。

    • 其他对 gc.collect() 的调用保持不变。

  • locale.nl_langinfo() 函数现在会在某些情况下临时设置 LC_CTYPE 区域设置。这种临时修改会影响其他线程。(由 Serhiy Storchaka 在 gh-69998 中贡献。)

  • types.UnionType 现在是 typing.Union 的别名,这会导致部分行为发生变化。更多细节请参见 上文。(由 Jelle Zijlstra 在 gh-105499 中贡献。)

  • 注解的运行时行为在多个方面发生了变化;详见 上文。虽然大多数与注解交互的代码应该能继续正常工作,但某些未记录的细节可能会有不同的表现。

Changes in the C API

  • Py_Finalize() now deletes all interned strings. This is backwards incompatible to any C extension that holds onto an interned string after a call to Py_Finalize() and is then reused after a call to Py_Initialize(). Any issues arising from this behavior will normally result in crashes during the execution of the subsequent call to Py_Initialize() from accessing uninitialized memory. To fix, use an address sanitizer to identify any use-after-free coming from an interned string and deallocate it during module shutdown. (Contributed by Eddie Elizondo in gh-113601.)

  • Unicode异常对象 C API现会在异常参数不是 UnicodeError 对象时抛出 TypeError 异常。(由Bénédikt Tran在 gh-127691 中贡献)