interrupt() 函数来工作。该函数接受任何 JSON 可序列化值,该值将呈现给调用者。当您准备好继续时,通过使用 Command 重新调用图来恢复执行,然后该值将成为节点内部 interrupt() 调用的返回值。
与静态断点(在特定节点之前或之后暂停)不同,中断是动态的:它们可以放置在代码中的任何位置,并且可以根据应用程序逻辑有条件地触发。
- 检查点保存您的位置:检查点写入确切的图状态,因此即使处于错误状态,您也可以稍后恢复。
thread_id是您的指针:设置config={"configurable": {"thread_id": ...}}以告诉检查点器加载哪个状态。- 中断负载通过
chunk["interrupts"]呈现:当使用version="v2"流式传输时,您传递给interrupt()的值会出现在values流部分的interrupts字段中,因此您知道图正在等待什么。
thread_id 实际上是您的持久化游标。重用它将恢复相同的检查点;使用新值将启动一个具有空状态的全新线程。
使用 interrupt 暂停
interrupt 函数暂停图执行并向调用者返回一个值。当您在节点内调用 interrupt 时,LangGraph 保存当前图状态并等待您使用输入恢复执行。
要使用 interrupt,您需要:
- 一个检查点器来持久化图状态(在生产中使用持久的检查点器)
- 配置中的线程 ID,以便运行时知道从哪个状态恢复
- 在您想要暂停的位置调用
interrupt()(负载必须是 JSON 可序列化的)
interrupt 时,会发生以下情况:
- 图执行在调用
interrupt的确切点被挂起 - 状态被保存使用检查点器,以便稍后可以恢复执行。在生产中,这应该是一个持久的检查点器(例如,由数据库支持)
- 值被返回给调用者,位于
__interrupt__下;它可以是任何 JSON 可序列化值(字符串、对象、数组等) - 图无限期等待,直到您使用响应恢复执行
- 响应被传回节点,当您恢复时,成为
interrupt()调用的返回值
恢复中断
中断暂停执行后,您可以通过使用包含恢复值的Command 再次调用图来恢复图。恢复值被传回 interrupt 调用,允许节点使用外部输入继续执行。
- v2 (LangGraph >= 1.1)
- v1 (默认)
- 您必须使用与中断发生时相同的线程 ID 进行恢复
- 传递给
Command(resume=...)的值成为interrupt调用的返回值 - 恢复时,节点从调用
interrupt的节点开头重新启动,因此interrupt之前的任何代码将再次运行 - 您可以传递任何 JSON 可序列化值作为恢复值
常见模式
中断解锁的关键功能是暂停执行并等待外部输入。这对于各种用例非常有用,包括:- 审批工作流:在执行关键操作(API 调用、数据库更改、金融交易)之前暂停
- 处理多个中断:在单次调用中恢复多个中断时,将中断 ID 与恢复值配对
- 审查和编辑:让人类在继续之前审查和修改 LLM 输出或工具调用
- 中断工具调用:在执行工具调用之前暂停,以便在执行前审查和编辑工具调用
- 验证人类输入:在继续下一步之前暂停以验证人类输入
使用人机回环 (HITL) 中断流式传输
在构建具有人机回环工作流的交互式代理时,您可以同时流式传输消息块和节点更新,以在处理中断时提供实时反馈。 使用多个流模式("messages" 和 "updates")以及 subgraphs=True(如果存在子图)来:
- 实时流式传输 AI 响应
- 检测图何时遇到中断
- 无缝处理用户输入并恢复执行
version="v2":所有块都是具有type、ns和data键的StreamPart字典chunk["type"]:根据流模式("messages"、"updates"等)缩小范围以进行类型推断chunk["ns"]:标识源图(根图为空元组,子图则填充)subgraphs=True:在嵌套图中检测中断所必需Command(resume=...):使用用户提供的数据恢复图执行
处理多个中断
当并行分支同时中断时(例如,分发到多个节点,每个节点都调用interrupt()),您可能需要在单次调用中恢复多个中断。
当在单次调用中恢复多个中断时,将每个中断 ID 映射到其恢复值。
这确保每个响应在运行时与正确的中断配对。
批准或拒绝
中断最常见的用途之一是在关键操作之前暂停并请求批准。例如,您可能希望让人类批准 API 调用、数据库更改或任何其他重要决策。True 以批准或 False 以拒绝:
完整示例
完整示例
审查和编辑状态
有时您希望让人类在继续之前审查和编辑图状态的一部分。这对于纠正 LLM、添加缺失信息或进行调整非常有用。完整示例
完整示例
工具中的中断
您也可以直接在工具函数中放置中断。这使得工具本身在每次调用时暂停以请求批准,并允许在执行前对工具调用进行人类审查和编辑。 首先,定义一个使用interrupt 的工具:
完整示例
完整示例
验证人类输入
有时您需要验证来自人类的输入,并在无效时再次询问。您可以使用循环中的多个interrupt 调用来实现这一点。
完整示例
完整示例
中断规则
当您在节点内调用interrupt 时,LangGraph 通过引发异常来挂起执行,该异常向运行时发出暂停信号。此异常通过调用堆栈向上传播,并由运行时捕获,运行时通知图保存当前状态并等待外部输入。
当执行恢复(在您提供请求的输入后),运行时从头开始重新启动整个节点——它不会从调用 interrupt 的确切行恢复。这意味着 interrupt 之前的任何代码将再次执行。因此,在使用中断时,有一些重要的规则需要遵循,以确保它们按预期行为。
不要在 interrupt 调用中包装 try/except
interrupt 在调用点暂停执行的方式是抛出一个特殊的异常。如果您将 interrupt 调用包装在 try/except 块中,您将捕获此异常,中断将不会被传回图。
- ✅ 将
interrupt调用与易出错的代码分开 - ✅ 在 try/except 块中使用特定的异常类型
- 🔴 不要在裸 try/except 块中包装
interrupt调用
不要在节点内重新排序 interrupt 调用
在单个节点中使用多个中断很常见,但如果处理不当,可能会导致意外行为。
当节点包含多个中断调用时,LangGraph 会为执行节点的任务维护一个特定的恢复值列表。每当执行恢复时,它都会从节点的开头开始。对于遇到的每个中断,LangGraph 会检查任务的恢复列表中是否存在匹配值。匹配是严格基于索引的,因此节点内中断调用的顺序很重要。
- ✅ 保持
interrupt调用在节点执行之间一致
不要在 interrupt 调用中返回复杂值
根据使用的检查点器,复杂值可能无法序列化(例如,您无法序列化函数)。为了使您的图适应任何部署,最好只使用可以合理序列化的值。
- ✅ 传递简单、JSON 可序列化的类型给
interrupt - ✅ 传递具有简单值的字典/对象
- 🔴 不要将函数、类实例或其他复杂对象传递给
interrupt
在 interrupt 之前调用的副作用必须是幂等的
因为中断通过重新运行它们被调用的节点来工作,所以在 interrupt 之前调用的副作用应该(理想情况下)是幂等的。就上下文而言,幂等性意味着相同的操作可以应用多次,而不会改变初始执行之外的结果。
例如,您可能在节点内有一个更新记录的 API 调用。如果在调用之后调用 interrupt,则在节点恢复时,它将被多次重新运行,可能会覆盖初始更新或创建重复记录。
- 🔴 不要在
interrupt之前执行非幂等操作 - 🔴 不要在不检查是否存在的情况下创建新记录
与作为函数调用的子图一起使用
当在节点内调用子图时,父图将从调用子图并触发interrupt 的节点的开头恢复执行。类似地,子图也将从调用 interrupt 的节点的开头恢复。
使用中断调试
要调试和测试图,您可以使用静态中断作为断点,以逐步执行图执行,一次一个节点。静态中断在节点执行之前或之后的定义点触发。您可以在编译图时通过指定interrupt_before 和 interrupt_after 来设置这些。
静态中断不推荐用于人机回环工作流。请改用
interrupt 函数。- 编译时
- 运行时
- 断点在
compile时设置。 interrupt_before指定在节点执行之前应暂停执行的节点。interrupt_after指定在节点执行之后应暂停执行的节点。- 需要检查点器才能启用断点。
- 图运行直到命中第一个断点。
- 通过为输入传递
None来恢复图。这将运行图直到命中下一个断点。
使用 LangSmith Studio
您可以使用 LangSmith Studio 在 UI 中设置图中的静态中断,然后再运行图。您还可以使用 UI 在执行的任何点检查图状态。

