
代理服务器自动处理检查点保存
使用 代理服务器 时,您无需手动实现或配置检查点保存器。服务器会在后台为您处理所有持久化基础设施。
为什么使用持久化
以下功能需要持久化:- 人在回路:检查点保存器通过允许人类检查、中断和批准图步骤来促进 人在回路工作流。这些工作流需要检查点保存器,因为人员必须能够在任何时间点查看图的状态,并且图必须能够在人员对状态进行任何更新后恢复执行。有关示例,请参阅 中断。
- 记忆:检查点保存器允许在交互之间存在 “记忆”。在重复的人类交互(如对话)的情况下,任何后续消息都可以发送到该线程,该线程将保留其对先前交互的记忆。有关如何使用检查点保存器添加和管理对话记忆的信息,请参阅 添加记忆。
- 时间旅行:检查点保存器允许 “时间旅行”,允许用户重放先前的图执行以审查和/或调试特定的图步骤。此外,检查点保存器使得可以在任意检查点处分叉图状态,以探索替代轨迹。
- 容错性:检查点保存提供了容错性和错误恢复:如果一个或多个节点在给定的超级步骤中失败,您可以从最后一个成功的步骤重新启动图。
- 待处理写入:当图节点在给定的 超级步骤 中执行失败时,LangGraph 会存储在该超级步骤中成功完成的任何其他节点的待处理检查点写入。当您从该超级步骤恢复图执行时,您不会重新运行成功的节点。
核心概念
线程
线程是分配给检查点保存器保存的每个检查点的唯一 ID 或线程标识符。它包含一系列 运行 的累积状态。当运行执行时,助手底层图的 状态 将被持久化到线程中。 当使用检查点保存器调用图时,您必须在配置的configurable 部分指定 thread_id:
thread_id 作为存储和检索检查点的主键。没有它,检查点保存器无法保存状态或在 中断 后恢复执行,因为检查点保存器使用 thread_id 来加载保存的状态。
检查点
线程在特定时间点的状态称为检查点。检查点是在每个 超级步骤 保存的图状态快照,由StateSnapshot 对象表示(有关完整字段参考,请参阅 StateSnapshot 字段)。
超级步骤
LangGraph 在每个 超级步骤 边界创建一个检查点。超级步骤是图的一个单独“滴答”,其中为该步骤调度的所有节点执行(可能并行)。对于像START -> A -> B -> END 这样的顺序图,输入、节点 A 和节点 B 有单独的超级步骤——在每个之后产生一个检查点。理解超级步骤边界对于 时间旅行 很重要,因为您只能从检查点(即超级步骤边界)恢复执行。
检查点被持久化,可用于在以后恢复线程的状态。
让我们看看当按如下方式调用简单图时保存了哪些检查点:
- 空检查点,下一个要执行的节点是
START - 包含用户输入
{'foo': '', 'bar': []}的检查点,下一个要执行的节点是nodeA - 包含
nodeA输出{'foo': 'a', 'bar': ['a']}的检查点,下一个要执行的节点是nodeB - 包含
nodeB输出{'foo': 'b', 'bar': ['a', 'b']}的检查点,没有下一个要执行的节点
bar 通道值包含来自两个节点的输出,因为我们为 bar 通道设置了一个归约器。
检查点命名空间
每个检查点都有一个checkpoint_ns(检查点命名空间)字段,用于标识它属于哪个图或子图:
""(空字符串):检查点属于父(根)图。"node_name:uuid":检查点属于作为给定节点调用的子图。对于嵌套子图,命名空间用|分隔符连接(例如"outer_node:uuid|inner_node:uuid")。
获取和更新状态
获取状态
与保存的图状态交互时,您必须指定 线程标识符。您可以通过调用graph.getState(config) 来查看图的_最新_状态。这将返回一个 StateSnapshot 对象,该对象对应于与配置中提供的线程 ID 关联的最新检查点,或者如果提供了检查点 ID,则对应于与该检查点 ID 关联的检查点。
getState 的输出将如下所示:
StateSnapshot 字段
| 字段 | 类型 | 描述 |
|---|---|---|
values | object | 此检查点处的状态通道值。 |
next | string[] | 下一个要执行的节点名称。空 [] 表示图已完成。 |
config | object | 包含 thread_id、checkpoint_ns 和 checkpoint_id。 |
metadata | object | 执行元数据。包含 source("input"、"loop" 或 "update")、writes(节点输出)和 step(超级步骤计数器)。 |
createdAt | string | 此检查点创建时的 ISO 8601 时间戳。 |
parentConfig | object | null | 上一个检查点的配置。第一个检查点为 null。 |
tasks | PregelTask[] | 此步骤要执行的任务。每个任务都有 id、name、error、interrupts,以及可选的 state(子图快照,当使用 subgraphs: true 时)。 |
获取状态历史记录
您可以通过调用graph.getStateHistory(config) 获取给定线程的完整图执行历史记录。这将返回与配置中提供的线程 ID 关联的 StateSnapshot 对象列表。重要的是,检查点将按时间顺序排列,最新的检查点/StateSnapshot 是列表中的第一个。
getStateHistory 的输出将如下所示:

查找特定检查点
您可以过滤状态历史记录以查找符合特定条件的检查点:重放
重放会重新执行来自先前检查点的步骤。使用先前的checkpoint_id 调用图以重新运行该检查点之后的节点。检查点之前的节点被跳过(它们的结果已保存)。检查点之后的节点重新执行,包括任何 LLM 调用、API 请求或 中断——在重放期间总是会重新触发。
有关重放过去执行的完整详细信息和代码示例,请参阅 时间旅行。

更新状态
您可以使用graph.updateState() 编辑图状态。这会创建一个具有更新值的新检查点——它不会修改原始检查点。更新的处理方式与节点更新相同:当定义了 归约器 函数时,值会通过这些函数传递,因此带有归约器的通道会_累积_值而不是覆盖它们。
您可以选择指定 asNode 来控制更新被视为来自哪个节点,这会影响接下来执行哪个节点。有关详细信息,请参阅 时间旅行:asNode。

内存存储

Store 接口。作为说明,我们可以定义一个 InMemoryStore 来跨线程存储用户信息。我们只需像以前一样使用检查点保存器编译图,并传递存储。
LangGraph API 自动处理存储
使用 LangGraph API 时,您无需手动实现或配置存储。API 会在后台为您处理所有存储基础设施。
InMemoryStore 适用于开发和测试。对于生产环境,请使用持久化存储,如
PostgresStore、MongoDBStore 或 RedisStore。所有实现都扩展了 BaseStore,这是在节点函数签名中使用的类型注解。基本用法
首先,让我们在不使用 LangGraph 的情况下单独展示这一点。tuple 进行命名空间划分,在这个特定示例中,它将是 (<user_id>, "memories")。命名空间可以是任意长度并表示任何内容,不一定是特定于用户的。
store.put 方法将记忆保存到存储中的命名空间。执行此操作时,我们指定如上定义的命名空间,以及记忆的键值对:键只是记忆的唯一标识符(memory_id),值(字典)是记忆本身。
store.search 方法读取命名空间中的记忆,该方法将返回给定用户的所有记忆作为列表。最近的记忆是列表中的最后一个。
-
value:此记忆的值 -
key:此记忆在此命名空间中的唯一键 -
namespace:字符串元组,此记忆类型的命名空间虽然类型是tuple,但在转换为 JSON 时可能会序列化为列表(例如['1', 'memories'])。 -
createdAt:此记忆创建时的时间戳 -
updatedAt:此记忆更新时的时间戳
语义搜索
除了简单的检索,存储还支持语义搜索,允许您根据含义而不是精确匹配来查找记忆。要启用此功能,请使用嵌入模型配置存储:fields 参数或在存储记忆时指定 index 参数来控制记忆的哪些部分被嵌入:
在 LangGraph 中使用
有了这一切,我们在 LangGraph 中使用memoryStore。memoryStore 与检查点保存器协同工作:检查点保存器将状态保存到线程中,如上所述,而 memoryStore 允许我们存储任意信息以供_跨_线程访问。我们如下编译图,同时使用检查点保存器和 memoryStore。
thread_id 调用图,同时也使用 user_id,我们将用它来将记忆命名空间划分到这个特定用户,如上所示。
runtime 参数在_任何节点_中访问存储和 userId。以下是如何使用它来保存记忆的示例:
store.search 方法获取记忆。回想一下,记忆作为对象列表返回,可以转换为字典。
user_id 相同,我们仍然可以访问相同的记忆。
langgraph.json 文件中配置索引设置。例如:
优化检查点存储
检查点保存器库
在底层,检查点保存由符合BaseCheckpointSaver 接口的检查点保存器对象提供支持。LangGraph 提供了多种检查点保存器实现,所有实现都通过独立的、可安装的库实现。
@langchain/langgraph-checkpoint:检查点保存器保存器的基础接口(BaseCheckpointSaver)和序列化/反序列化接口(SerializerProtocol)。包括用于实验的内存检查点保存器实现(MemorySaver)。LangGraph 自带@langchain/langgraph-checkpoint。@langchain/langgraph-checkpoint-sqlite:使用 SQLite 数据库的 LangGraph 检查点保存器实现(SqliteSaver)。适用于实验和本地工作流。需要单独安装。@langchain/langgraph-checkpoint-postgres:使用 Postgres 数据库的高级检查点保存器(PostgresSaver),在 LangSmith 中使用。适用于生产环境。需要单独安装。@langchain/langgraph-checkpoint-mongodb:由 MongoDB 支持的高级检查点保存器(MongoDBSaver)和长期记忆存储(MongoDBStore)。存储支持跨线程持久化,并可选集成向量搜索。适用于生产环境。需要单独安装。@langchain/langgraph-checkpoint-redis:使用 Redis 数据库的高级检查点保存器(RedisSaver)。适用于生产环境。需要单独安装。
检查点保存器接口
每个检查点保存器都符合BaseCheckpointSaver 接口并实现以下方法:
.put- 存储检查点及其配置和元数据。.putWrites- 存储与检查点关联的中间写入(即 待处理写入)。.getTuple- 使用给定配置(thread_id和checkpoint_id)获取检查点元组。这用于填充graph.getState()中的StateSnapshot。.list- 列出与给定配置和过滤条件匹配的检查点。这用于填充graph.getStateHistory()中的状态历史记录。
将这些文档连接到 Claude、VSCode 等,通过 MCP 获取实时答案。

