惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

SecWiki News
SecWiki News
H
Help Net Security
罗磊的独立博客
Stack Overflow Blog
Stack Overflow Blog
M
MIT News - Artificial intelligence
Jina AI
Jina AI
L
LangChain Blog
K
Kaspersky official blog
I
Intezer
Martin Fowler
Martin Fowler
爱范儿
爱范儿
AWS News Blog
AWS News Blog
The Hacker News
The Hacker News
Recorded Future
Recorded Future
人人都是产品经理
人人都是产品经理
H
Hackread – Cybersecurity News, Data Breaches, AI and More
C
CXSECURITY Database RSS Feed - CXSecurity.com
Spread Privacy
Spread Privacy
Simon Willison's Weblog
Simon Willison's Weblog
U
Unit 42
N
News and Events Feed by Topic
A
Arctic Wolf
G
GRAHAM CLULEY
Microsoft Azure Blog
Microsoft Azure Blog
博客园 - 聂微东
F
Fortinet All Blogs
C
Cisco Blogs
美团技术团队
Vercel News
Vercel News
K
KPMG report finds enterprise disconnect between AI and its ROI | CIO
H
Hacker News: Front Page
T
Tailwind CSS Blog
I
InfoQ
宝玉的分享
宝玉的分享
Google DeepMind News
Google DeepMind News
博客园 - 司徒正美
P
Palo Alto Networks Blog
A
About on SuperTechFans
Threat Intelligence Blog | Flashpoint
Threat Intelligence Blog | Flashpoint
云风的 BLOG
云风的 BLOG
TaoSecurity Blog
TaoSecurity Blog
Google Online Security Blog
Google Online Security Blog
Exploit-DB.com RSS Feed
Exploit-DB.com RSS Feed
P
Privacy & Cybersecurity Law Blog
H
Heimdal Security Blog
cs.CV updates on arXiv.org
cs.CV updates on arXiv.org
Hacker News: Ask HN
Hacker News: Ask HN
O
OpenAI News
博客园 - Franky
Scott Helme
Scott Helme

博客园_首页

Plist 二进制格式 Milvus 和 PGVector,哪个更好? OpenClaw 已过时?在 VS Code 中运行 Hermes Agent! 第30篇文章:一个大三计科生的自白 Manim如何在数学公式中完美显示中文? Docker 部署 RocketMQ 5 并发编程核心概念辨析 C#事务处理最佳实践:别再让“主表存了、明细丢了”的破事发生 CLI 是什么?为什么大厂突然集体卷命令行? 【从0到1构建一个ClaudeAgent】协作-自主Agent UIImageView 设置图片不生效的原因排查 最小二乘问题详解20:无先验约束下的增量式SFM自由网平差 痞子衡嵌入式:大话双核i.MXRT1180之XIP应用里借助MU实现可靠Flash IAP的方法 AI Chat 封装, SemanticKerne.AiProvider.Unified 已发布 Windows下右键编辑js文件无法打开记事本——在注册表中使用环境变量 在后台服务中使用 Scoped 服务,为什么总是报错? H200 安装驱动并使用sglang启动模型 wireshark 抓包Trap上报告警内容 我用 AI 辅助开发了一系列小工具(2):图片压缩工具 [A Primer On MC and CC] 2.1 Memory Consistency 1 - 指令重排序和 SC 模型 Oracle数据库SCN推进技术详解与实践指南 玩转控件:封装个带图片的Label控件 Claude Code 4.7 真正该升级的不是模型,而是你的工作流 前端小白一句话,AI 帮我做了个颜值拉满的桌面媒体播放器。当代码不再是门槛,一句话编程就是现实。 5. WorkBuddy: 小龙虾的灵魂三件套,让你的小龙虾不只是工具 SQLite 分片方案实战:三种分片策略的深度对比 告别简陋 UI!一款基于 Fluent Design 和基于 WinUI 的开源免费、现代化的 Avalonia UI 控件库 关于二进制排列组合枚举的总结 AI开发-python-LangGraph框架(3-27-LangGraph从零实现大模型智能决策工作流) ElasticSearch主分片和副本分片概念详解 【002】HTTPS 粗解:证书、TLS 握手与对后端配置的影响 Hermes Agent 一周暴涨五万 Star,但我劝你别急着追 明明连接的是Redis的DB0,为什么能查到DB3的数据? 【从0到1构建一个ClaudeAgent】协作-Agent团队 熟悉电子元器件之后,电子小白下一步该怎么走? MAF快速入门(23)通过C#类定义Skills .NET 高级开发 | 手写一个对象映射框架 FastAPI数据库ORM怎么选?我肝了三个Demo后,终于不再纠结了 mysqldump 参数拾遗:在遗忘与铭记之间 C# .NET 周刊|2026年3月5期 Claude code入门 - 陈彦斌 一文学习入门 ThingsBoard 开源物联网平台 GitHub 热门项目 | 2026年04月16日 如何为GIT设置全局勾子,为每次提交追加信息 Number.isFinite和isFinite与isNaN()和Number.isNaN的区别 PortSwigger SQL注入LAB2 推荐一个测试人必备的Skills,从功能到性能全搞定(附详细实操和安装下载方式) 筑基期:掌握Odoo基础核心知识点02(Odoo XML 开发方式详解) GLM模型这么火,咱们用vllm也咧一个呗! 深入理解 AbortController:从底层原理到跨语言设计哲学 字符串学习笔记 多租户系统框架的基础模块设计和分析设计 Apache SeaTunnel Zeta 为什么能做到“又快又稳”? AI开发-python-LangGraph框架(3-26-LangGraph基本概念及第一个简单样例) Vue 3 组件通信,别只会用 Props 和 Emits 了,这几个狠活儿你得看看 ElasticSearch7.X版本配置密码 用Manim实现动态交点计算--从一个动点问题说起 团结引擎+Addressable+Instant Game打包抖音小游戏 function call 实战:让 LLM 自动判断 pod 异常、调用日志工具并完成故障分析 bubseek —— 让 Agent 的足迹,变成团队的洞察 通过 C# 读取并导出 PDF 书签 如何用 GitHub Actions 实现 Steam 自动化发布 【从0到1构建一个ClaudeAgent】并发-后台任务 .NET 高级开发 | 定制 ASP.NET Core 框架 电子小白:什么是运算放大器(运放) zero2Agent:面向大厂面试的 Agent 工程教程,从概念到生产的完整学习路线 堆上的ORW HC32F460 USB CDC通信异常:非对齐访问异常排查 20260413-Hyperbridge 攻击事件:发生在默克尔山上的验证绕过 那些喊着AI 要淘汰你的人,正在靠你的焦虑赚大钱! 深度学习进阶(八)Swin Transformer 最小二乘问题详解19:带先验约束的增量式SFM优化与实现 SnapTranslate 3.0 正式发布:全局划词翻译 + 完整英语学习闭环,一站式搞定查词、记词、复习 工作的意义、工作的困难认知再思考 .NET + AI 进阶实战:基于类的技能开发 - 打造可治理的 Agent 能力模块 【从0到1构建一个ClaudeAgent】规划与协调-技能 上周热点回顾(4.6-4.12) 电子小白的工具三件套:面包板、杜邦线、万能板 单表五亿数据的查询优化 | Mysql、StarRocks 2. WorkBuddy:从“我是谁”到“帮我干活” C# 如何减少代码运行时间:7 个实战技巧 基于HelixToolkit.SharpDX 渲染3D模型 - 笺上知微 从零开始的双臂具身VLA起源及现阶段发展综述 - SkyXZ 记对 xonsh shell 的使用, 脚本编写, 迁移及调优 - pluvium27 受够了Vibe Coding的失控?换个起点,让AI事半功倍 从开始配置漏洞环境到漏洞复现流程 - 難しい 关于10年工作经验的程序员对OpenClaw的实战经验分享以及看法 - 虚无境 Any metadata 的内存布局 C# .NET 周刊|2026年3月2期 - InCerry 我帮你测过了,测试圈排名第二的 Skill 依然很牛逼 Skill Discovery | 无监督技能发现的经典工作总结 - MoonOut 上下文工程是什么?过时了么?一文讲明白! - 一枫说码 开了 TUN 模式还是直连?90% 的人都踩过这个坑 AScript扩展多种脚本语言 - rockey627 AI 学习笔记:Agent 的记忆机制 你能被装进一个文件里吗?——7 万人把同事"蒸馏"成了 AI - 我没有三颗心脏 Claude Code 通关手册(七):给 AI 装上技能包——Skills 完全指南 - 暮色之狐 在浏览器中快速编辑代码:VSCode Web 集成实践 - Newbe36524 蒸馏自己 skill?基于 Deepseek 的蒸馏器,丐版蒸馏方式,简单便捷 - To_Carpe_Diem Spring AI Aliababa和AgentScope,哪个更好? - 苏三说技术
28. Agent 执行到一半想暂停?用 interrupt 给它设个“关卡“!
老陈说编程 · 2026-06-18 · via 博客园_首页

Agent 跑起来之后,大多数时候我们希望它一气呵成把活干完。但总有些场景不太一样——比如 Agent 要调用一个会删文件的工具,你总得让人确认一下再动手吧?LangGraph 的 interrupt 机制就是干这个的:在指定节点前或节点后暂停执行,等人给了信号再接着跑。

动画视频在《28. Agent 执行到一半想暂停?用 interrupt 给它设个"关卡"!》

image

 interrupt_before:工具执行前拦截

最常用的场景是 interrupt_before。把它设为 ["tools"],意思是 Agent 调用完模型、决定要用哪个工具之后,先别急着执行,停下来等人发话。配置方式很简单,在 create_agent 里加一个参数就行:

agent = create_agent(
    model=model,
    tools=[calculate, write_file, read_file, list_dir],
    system_prompt="你是一个助手,会用工具计算、读写文件、列出目录。",
    debug=True,
    checkpointer=checkpointer,
    interrupt_before=["tools"]
)

这里有个前提——必须同时传入 checkpointer。原因很直接:暂停之后要恢复,就得靠 checkpointer 把当前状态存下来,不然中断 了就回不来了。

发送消息,观察暂停效果

配好之后,用 agent.invoke() 发一条消息试试:

config = {"configurable": {"thread_id": "session-2"}}
q = "计算 2024*12+500"
result = agent.invoke({"messages": [{"role": "user", "content": q}]}, config=config)

运行之后你会发现,Agent 调用模型产生了工具调用请求,但并没有真正执行工具——它停在了 tools 节点之前。result["messages"] 里的最后一条消息,就是 AI 生成的工具调用请求,你可以把它打印出来让用户看看 Agent 接下来打算干什么:

print(f"Agent 已暂停,最后消息: {result['messages'][-1].content[:100]}...")

Command(resume=...) 恢复执行

暂停不是目的,关键是暂停之后怎么继续。LangGraph 提供了一个 Command 对象,通过它的 resume 参数告诉 Agent"可以走了":

from langgraph.types import Command
# 恢复执行
result = agent.invoke(Command(resume="approved"), config=config)
print(f"答:{result['messages'][-1].content}")

注意 thread_id 不用换——因为 checkpointer 靠它找到上次暂停时的状态,从断点 处继续往下走。

如果把这个过程做成交互式的,就是让用户自己决定要不要放行:

# 暂停后等待用户输入
user_input = input("确认执行?(yes/no): ")
if user_input.lower() == "yes":
    result = agent.invoke(Command(resume=user_input), config=config)
    # 找到最后一条有内容的消息
    for msg in reversed(result['messages']):
        if isinstance(msg, ToolMessage):
            print(f"答:{msg.content}")
            break
else:
    print("已取消")

用户输入 yes,Agent 就继续调用工具、拿结果、生成回答;输入别的内容,这次工具调用就直接丢弃,流程到此结束。

interrupt_after:工具执行后再拦截

刚才说的是"事前审批"——工具还没跑,先问问人。那如果想在工具跑完之后再加一道关卡呢?把参数从 interrupt_before 换成 interrupt_after 就行,其他代码完全不用动:

agent = create_agent(
    model=model,
    tools=[calculate, write_file, read_file, list_dir],
    system_prompt="你是一个助手,会用工具计算、读写文件、列出目录。",
    debug=True,
    checkpointer=checkpointer,
   interrupt_after =["tools"]
)

一字之差,执行时机完全不同。interrupt_before 是 Agent 想好要调用什么工具了、但还没动手的时候拦截;interrupt_after 是工具已经跑完了、结果也拿到了,但 Agent 还没把结果送回模型做下一步推理的时候拦截。前者适合"你确定要这么做吗",后者适合"做完了,你看看结果对不对"。

恢复的方式一模一样:

user_input = input("工具已执行完毕,确认继续?(yes/no): ")
if user_input.lower() == "yes":
    result = agent.invoke(Command(resume=user_input), config=config)
    # 找到最后一条有内容的消息
    for msg in reversed(result['messages']):
        if isinstance(msg, ToolMessage):
            print(f"答:{msg.content}")
            break
else:
    print("已取消")

把上面的内容串起来,下面是一份可以直接运行的完整代码:

import os
import sqlite3
from dotenv import load_dotenv
from langchain.agents import create_agent, AgentState
from langchain.agents.middleware import AgentMiddleware
from langchain.chat_models import init_chat_model
from langchain_classic.agents import Agent
from langchain_community.tools import WriteFileTool, ReadFileTool, ListDirectoryTool
from langchain_core.messages import AIMessage, ToolMessage
from langchain_core.tools import tool, BaseTool
from langgraph.checkpoint.sqlite import SqliteSaver
from langgraph.runtime import Runtime
from langgraph.types import Command
 
load_dotenv()
prefix = "QWEN"
model = init_chat_model(
    model_provider="openai",
    configurable_fields=["model", "api_key", "base_url"],
    config_prefix=prefix
).with_config({
    "configurable": {
        f"{prefix}_model": os.getenv(f"{prefix}_MODEL"),
        f"{prefix}_api_key": os.getenv(f"{prefix}_API_KEY"),
        f"{prefix}_base_url": os.getenv(f"{prefix}_BASE_URL")
    }
})
 
 
class CalculateTool(BaseTool):
    name: str = "calculate"
    description: str = "计算数学表达式的值"
 
    def _run(self, expression: str) -> str:
        try:
            return f"计算结果:{eval(expression)}"
        except Exception as e:
            return f"计算错误:{str(e)}"
 
    async def _arun(self, expression: str) -> str:
        return self._run(expression)
 
 
calculate = CalculateTool()
write_file = WriteFileTool()
read_file = ReadFileTool()
list_dir = ListDirectoryTool()
checkpoint_conn = sqlite3.connect("agent.db", check_same_thread=False, isolation_level=None)
checkpointer = SqliteSaver(checkpoint_conn)
 
agent = create_agent(
    model=model,
    tools=[calculate, write_file, read_file, list_dir],
    system_prompt="你是一个助手,会用工具计算、读写文件、列出目录。",
    debug=True,
    checkpointer=checkpointer,
    interrupt_before=["tools"]
)
config = {"configurable": {"thread_id": "session-2"}}
 
q = "计算 2024*12+500"
result = agent.invoke({"messages": [{"role": "user", "content": q}]}, config=config)
print(f"Agent 已暂停,最后消息: {result['messages'][-1].content[:100]}...")
# 暂停后等待用户输入
user_input = input("确认执行?(yes/no): ")
if user_input.lower() == "yes":
    result = agent.invoke(Command(resume=user_input), config=config)
    # 找到最后一条有内容的消息
    for msg in reversed(result['messages']):
        if isinstance(msg, ToolMessage):
            print(f"答:{msg.content}")
            break
else:
    print("已取消")
 
checkpoint_conn.close()

运行这份代码,你会看到这样的流程:发送问题后,Agent 调用模型、决定使用 calculate 工具,但在执行前暂停;控制台打印出 Agent 打算执行的操作,等待你输入;输入 yes,Agent 继续执行工具调用,计算结果、写文件,最后返回完整回答;输入其他内容,这次调用直接取消。

如果要把暂停点改到工具执行之后,只需要把 interrupt_before=["tools"] 改成 interrupt_after=["tools"],其他代码一行不用动。

interrupt_before 适合"事前审批",工具还没跑,先看看 Agent 想干嘛;interrupt_after 适合"事后审查",工具跑完了,检查一下结果再继续。两种机制配合使用,就能把 Agent 的执行流程牢牢攥在手里。