<?xml version="1.0" encoding="UTF-8"?>
<rss version="2.0"
	xmlns:content="http://purl.org/rss/1.0/modules/content/"
	xmlns:wfw="http://wellformedweb.org/CommentAPI/"
	xmlns:dc="http://purl.org/dc/elements/1.1/"
	xmlns:atom="http://www.w3.org/2005/Atom"
	xmlns:sy="http://purl.org/rss/1.0/modules/syndication/"
	xmlns:slash="http://purl.org/rss/1.0/modules/slash/"
	>

<channel>
	<title>潘锦的空间 &#187; Memory System</title>
	<atom:link href="https://www.phppan.com/tag/memory-system/feed/" rel="self" type="application/rss+xml" />
	<link>https://www.phppan.com</link>
	<description>SaaS SaaS架构 团队管理 技术管理 技术架构 PHP 内核 扩展 项目管理</description>
	<lastBuildDate>Sat, 25 Apr 2026 00:56:17 +0000</lastBuildDate>
	<language>zh-CN</language>
		<sy:updatePeriod>hourly</sy:updatePeriod>
		<sy:updateFrequency>1</sy:updateFrequency>
	<generator>https://wordpress.org/?v=3.9.40</generator>
	<item>
		<title>聊下 OpenClaw 的记忆系统</title>
		<link>https://www.phppan.com/2026/03/openclaw-memory/</link>
		<comments>https://www.phppan.com/2026/03/openclaw-memory/#comments</comments>
		<pubDate>Sun, 15 Mar 2026 11:37:01 +0000</pubDate>
		<dc:creator><![CDATA[admin]]></dc:creator>
				<category><![CDATA[架构和远方]]></category>
		<category><![CDATA[Agent]]></category>
		<category><![CDATA[Memory System]]></category>
		<category><![CDATA[OpenClaw]]></category>

		<guid isPermaLink="false">https://www.phppan.com/?p=2477</guid>
		<description><![CDATA[OpenClaw 是最近 AI 圈最火的一个开源项目，没有之一。 从去年的 Agent 年，到今年的 AI 个 [&#8230;]]]></description>
				<content:encoded><![CDATA[<section id="nice" data-tool="mdnice编辑器" data-website="https://www.mdnice.com">
<p data-tool="mdnice编辑器">OpenClaw 是最近 AI 圈最火的一个开源项目，没有之一。</p>
<p data-tool="mdnice编辑器">从去年的 Agent 年，到今年的 AI 个人助理，OpenClaw 和去年 Manus 一样的，爆到不行，而且还是开源的版本。</p>
<p data-tool="mdnice编辑器">由于最近自己也在做 Agent，于是也看了 OpenClaw 的代码来了解其记忆系统的实现。有一些觉得可以借鉴学习的地方。</p>
<p data-tool="mdnice编辑器">OpenClaw 的记忆系统其实比较简单：它把「记忆」拆成了<strong>文件</strong>、<strong>索引</strong>、<strong>召回注入</strong>。</p>
<h1 data-tool="mdnice编辑器"><span class="content">1. 「Agent 记忆系统」的定义</span></h1>
<p data-tool="mdnice编辑器">在 Agent 工程里，记忆是一套能力组合：</p>
<ol data-tool="mdnice编辑器">
<li>
<section><strong>持久化</strong>：跨会话保存事实、偏好、决策、未完成事项。</section>
</li>
<li>
<section><strong>可检索</strong>：能在需要时把相关片段拉出来，且可控预算。</section>
</li>
<li>
<section><strong>可注入</strong>：把召回结果以确定的结构进入模型上下文，不靠「它自己想起来」。</section>
</li>
<li>
<section><strong>可审计</strong>：出了错能定位「写入发生在什么时候」「召回命中了什么」「注入了哪些行」。</section>
</li>
<li>
<section><strong>可治理</strong>：能处理泄露风险、过期信息、重复信息、冲突信息。</section>
</li>
</ol>
<p data-tool="mdnice编辑器">OpenClaw 的实现路径非常「工程」：<strong>Markdown 作为事实源</strong>，<strong>SQLite 作为检索索引</strong>，<strong>toolResult 作为注入通道</strong>。</p>
<h1 data-tool="mdnice编辑器"><span class="content">2. OpenClaw 的三层记忆</span></h1>
<p data-tool="mdnice编辑器">OpenClaw 的记忆从存储逻辑上来看可以分为三层：</p>
<h2 data-tool="mdnice编辑器"><span class="content">2.1 会话记忆</span></h2>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>介质</strong>：内存为主，但 OpenClaw 会把 session 打印成类似日志的文件，放到 <code>sessions</code> 目录。</section>
</li>
<li>
<section><strong>内容</strong>：用户消息、OpenClaw 的思考过程、工具调用、skill 调用、最终回复。</section>
</li>
<li>
<section><strong>边界</strong>：会话结束后「可用性」就不可靠了。你能在文件里回放，但模型下一次对话并不会天然带着它。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">会话记忆更像「trace」。我们不能指望它解决跨会话连续性，只用它做排障、复盘、抽取素材（写入短期/长期）。</p>
<h2 data-tool="mdnice编辑器"><span class="content">2.2 短期记忆</span></h2>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>介质</strong>：磁盘，<code>memory/YYYY-MM-DD.md</code> 为主（参考内容给了例子 <code>2026-03-10.md</code>）。</section>
</li>
<li>
<section><strong>内容</strong>：当天重要事件、过程笔记、TODO。关键点是「重要性」由人设与调教决定。</section>
</li>
<li>
<section><strong>边界</strong>：短期记忆是<strong>追加式日志</strong>，质量会漂移。写得越多，噪声越大；但写得太少，又召回不到。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">短期记忆适合承接「会话压缩之前的落盘」和「跨几天的上下文连续性」。它不是最终事实源，别把它当永久协议文档。</p>
<h2 data-tool="mdnice编辑器"><span class="content">2.3 长期记忆</span></h2>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>介质</strong>：工作区根目录 <code>MEMORY.md</code>（参考内容明确）。</section>
</li>
<li>
<section><strong>内容</strong>：核心认知与关系、偏好风格、长期目标、进行中任务、关键事件/教训/决策。</section>
</li>
<li>
<section><strong>边界</strong>：参考内容强调「只在主会话加载」，群聊等任务不加载，避免泄露。</section>
</li>
</ul>
<p data-tool="mdnice编辑器"><code>MEMORY.md</code> ==「可执行的组织记忆」。它的价值不在于「写得多」，在于「冲突少、可被召回、能约束后续行为」。这层要治理，要像维护配置一样维护。</p>
<h1 data-tool="mdnice编辑器"><span class="content">3. OpenClaw 的文件布局</span></h1>
<h2 data-tool="mdnice编辑器"><span class="content">3.1 「会话快照」文件</span></h2>
<p data-tool="mdnice编辑器">快照文件主要是解决 <code>/new</code>、<code>/reset</code> 指令的断片</p>
<p data-tool="mdnice编辑器">session-memory 的 Hook 会在你执行 <code>/new</code> 或 <code>/reset</code> 前，把上一会话最近 N 条对话（默认 15 条）抽出来，写成一个 Markdown 文件放到 <code>workspace/memory/</code> 下。</p>
<ul data-tool="mdnice编辑器">
<li>
<section>命名：<code>YYYY-MM-DD-&lt;slug&gt;.md</code>，slug 通常由 LLM 根据主题生成；LLM 不可用就回退成 <code>HHMM</code>。</section>
</li>
<li>
<section>关键点：这种文件也会被检索索引到。原因是 <code>listMemoryFiles()</code> 会递归扫描 <code>workspace/memory/</code> 下所有 <code>.md</code>，并不要求必须是 <code>YYYY-MM-DD.md</code>（ <code>src/memory/internal.ts:115-145</code> ）。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">这个机制对「人类工作流」很友好。很多团队的真实使用是：今天临时开了个话题，明天又忘了开在另一个会话里。会话快照能把碎片变成可检索素材，后面再沉淀进 <code>MEMORY.md</code>。</p>
<h2 data-tool="mdnice编辑器"><span class="content">3.2 <code>memory/main.sqlite</code>：索引库</span></h2>
<ul data-tool="mdnice编辑器">
<li>
<section><code>memory/main.sqlite</code> 基本可以确定是「记忆搜索（memory_search）」用的 SQLite 索引库。</section>
</li>
<li>
<section>索引对象：<code>MEMORY.md</code>、<code>memory/**/*.md</code>，以及你配置的 <code>extraPaths</code>，可选 session transcripts。</section>
</li>
<li>
<section>检索方式：FTS/BM25 关键词检索 +（可选）向量相似度检索（sqlite-vec）。</section>
</li>
<li>
<section>它存的典型结构：<code>files</code>、<code>chunks</code>、<code>embedding_cache</code>，以及可选的 FTS 表、向量虚表。</section>
</li>
</ul>
<p data-tool="mdnice编辑器"><strong>事实源是文本文件</strong>，<strong>索引是可重建的派生物</strong>。索引坏了你删掉重建就行；事实源坏了才是真的坏。</p>
<h1 data-tool="mdnice编辑器"><span class="content">4. 怎么建索引：</span></h1>
<p data-tool="mdnice编辑器">建索引我们关心的三件事：增量、去重、成本</p>
<p data-tool="mdnice编辑器">OpenClaw 的索引构建不是「每次全量重算」，也不是「精细 diff」：</p>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>更新粒度</strong>：按文件做增量。文件没变直接跳过。</section>
</li>
<li>
<section><strong>分块粒度</strong>：文件变了就重建该文件 chunks。</section>
</li>
<li>
<section><strong>向量成本</strong>：chunk embedding 通过缓存复用，避免重复调用 embedding provider。</section>
</li>
</ul>
<h2 data-tool="mdnice编辑器"><span class="content">4.1 扫描哪些文件会进索引</span></h2>
<p data-tool="mdnice编辑器">OpenClaw 会递归扫描 <code>workspace/memory/</code> 下所有 <code>.md</code>，并包含 <code>MEMORY.md</code>/<code>memory.md</code>。对应定位是 <code>src/memory/internal.ts:115-145</code>。</p>
<h2 data-tool="mdnice编辑器"><span class="content">4.2 文件级 hash：没变就跳过</span></h2>
<p data-tool="mdnice编辑器">文件 hash 的策略：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>Markdown：<code>hash = sha256(content)</code>（<code>internal.ts:L245-L263</code>）</section>
</li>
<li>
<section>多模态：buffer 也会 hash，最后把 <code>{path, contentText, mimeType, dataHash}</code> 做 JSON 再 sha256（<code>internal.ts:L204-L243</code>）</section>
</li>
<li>
<section>增量判定：对比 <code>files</code> 表里的 hash，一致就跳过（参考内容列了 memory 文件与 session 文件两条路径）。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">hash 是增量的核心。它的意义不止省时间，还省钱：embedding provider 往往是计费点。这种主要是对于使用第三方 embedding 的。</p>
<h2 data-tool="mdnice编辑器"><span class="content">4.3 分块策略：<code>tokens*4</code> 的字符近似</span></h2>
<p data-tool="mdnice编辑器"><code>chunkMarkdown()</code> 的策略：</p>
<ul data-tool="mdnice编辑器">
<li>
<section><code>maxChars = tokens * 4</code>，<code>overlapChars = overlap * 4</code></section>
</li>
<li>
<section>以「行」为主，超长行会被切段</section>
</li>
<li>
<section>每块生成 <code>hash=sha256(text)</code>，并有 <code>embeddingInput</code></section>
</li>
</ul>
<p data-tool="mdnice编辑器"><code>tokens*4</code> 这种近似在工程里挺常见，优点是简单、稳定、跨模型大差不差。缺点也明显：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>语言差异会影响 token/char 比例；中英文混排时 chunk 尺寸会漂。</section>
</li>
<li>
<section>以行切块对 Markdown 友好，但对「一行很长的 JSON 或日志」不友好。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">可以盯两个指标来调 chunking：</p>
<ol data-tool="mdnice编辑器">
<li>
<section>平均召回 snippet 的「可读性」和「自洽性」；</section>
</li>
<li>
<section>SQLite 体积与索引更新耗时。chunk 太小召回碎，太大注入贵。</section>
</li>
</ol>
<h2 data-tool="mdnice编辑器"><span class="content">4.4 chunk 的唯一标识与 upsert</span></h2>
<p data-tool="mdnice编辑器">去重靠 id，chunk 写入策略如下：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>chunk <code>id</code>：<code>sha256("${source}:${path}:${startLine}:${endLine}:${chunk.hash}:${provider.model}")</code></section>
</li>
<li>
<section>同一 id：<code>ON CONFLICT(id) DO UPDATE</code> 覆盖更新</section>
</li>
<li>
<section>文件要重建时会先清旧再写新（参考内容总结了「清旧再写新」语义）</section>
</li>
</ul>
<p data-tool="mdnice编辑器">这里有个很实际的 trade-off：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>它不做「chunk diff」，所以文件变了就重建该文件 chunks，逻辑简单，坏处是 IO 多。</section>
</li>
<li>
<section>但 embedding 通过缓存复用，把最贵的部分压下去了。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">另外，<code>id</code> 里带了 <code>provider.model</code>，这会带来一个工程后果：<strong>embedding 模型换了，chunk id 会变</strong>，索引层面等价于全量重建。 <code>provider/model/providerKey</code> 变化会触发 full reindex。</p>
<h2 data-tool="mdnice编辑器"><span class="content">4.5 embedding_cache</span></h2>
<p data-tool="mdnice编辑器">embedding 缓存的主键设计：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>主键：<code>(provider, model, provider_key, hash)</code></section>
</li>
<li>
<section><code>provider_key</code> 会把 endpoint/headers 等纳入指纹，避免跨配置污染缓存（而且会剔除授权头的细节在参考内容里提到）。</section>
</li>
<li>
<section>批量加载命中就跳过 embed；miss 才请求，成功回写缓存（对应 <code>manager-embedding-ops.ts</code> 的行段）。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">这是「上线能用」的关键。否则就会遇到一个很尴尬的情况：<br />
索引更新频繁触发 embedding 重算 → 延迟抖动 → API 费暴涨 → 还可能被 provider 限流。 system prompt 的 skills 段落甚至提醒「假设有 rate limits，避免 tight loop」，这就有点被打过之后写进规范的味道。</p>
<h1 data-tool="mdnice编辑器"><span class="content">5. 更新怎么触发</span></h1>
<p data-tool="mdnice编辑器">watch、interval、onSearch、session-delta</p>
<p data-tool="mdnice编辑器">索引更新如果做得「过勤」，会把 CPU 和 IO 吃满；做得「过懒」，召回就是过期的。OpenClaw 在触发上给了多条路径：</p>
<ol data-tool="mdnice编辑器">
<li>
<section><strong>watch</strong>：chokidar 监听 <code>MEMORY.md</code>、<code>memory.md</code>、<code>memory/**/*.md</code>（以及 extraPaths、多模态扩展），变更标记 dirty，debounce 后 <code>sync(reason="watch")</code>。</section>
</li>
<li>
<section><strong>interval</strong>：<code>sync.intervalMinutes&gt;0</code> 就 <code>setInterval</code> 定时跑。</section>
</li>
<li>
<section><strong>onSessionStart</strong>：search 前 <code>warmSession(sessionKey)</code>，若开了 <code>sync.onSessionStart</code>，每个 sessionKey 首次触发后台 sync。</section>
</li>
<li>
<section><strong>onSearch</strong>：search 时如果 dirty 且开了 <code>sync.onSearch</code>，后台触发 sync。</section>
</li>
<li>
<section><strong>session-delta</strong>：监听 transcript 更新，累计新增 bytes/lines，达到阈值后把相关 session 文件标脏，再 sync。</section>
</li>
</ol>
<p data-tool="mdnice编辑器">还有两点：</p>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>单飞锁</strong>：同一时刻只跑一个 sync，复用同一个 Promise（参考内容定位 <code>manager.ts:452-467</code>）。</section>
</li>
<li>
<section><strong>全量重建的原子 swap</strong>：写到 <code>.tmp-UUID</code>，完成后 swap（含 <code>wal/-shm</code>），避免半成品索引（参考内容定位 <code>manager-sync-ops.ts:1050-1158</code>）。</section>
</li>
</ul>
<h1 data-tool="mdnice编辑器"><span class="content">6. 召回是怎么发生的</span></h1>
<p data-tool="mdnice编辑器">主要看 <code>buildMemorySection()</code> 的代码，因为它把策略写死了：</p>
<pre class="custom" data-tool="mdnice编辑器"><code class="hljs">提示词：
<span class="hljs-keyword">function</span> buildSkillsSection(params: { skillsPrompt?: string; readToolName: string }) {
  const trimmed = params.skillsPrompt?.trim();
  <span class="hljs-keyword">if</span> (!trimmed) {
    <span class="hljs-built_in">return</span> [];
  }
  <span class="hljs-built_in">return</span> [
    <span class="hljs-string">"## Skills (mandatory)"</span>,
    <span class="hljs-string">"Before replying: scan &lt;available_skills&gt; &lt;description&gt; entries."</span>,
    `- If exactly one skill clearly applies: <span class="hljs-built_in">read</span> its SKILL.md at &lt;location&gt; with \`<span class="hljs-variable">${params.readToolName}</span>\`, <span class="hljs-keyword">then</span> follow it.`,
    <span class="hljs-string">"- If multiple could apply: choose the most specific one, then read/follow it."</span>,
    <span class="hljs-string">"- If none clearly apply: do not read any SKILL.md."</span>,
    <span class="hljs-string">"Constraints: never read more than one skill up front; only read after selecting."</span>,
    <span class="hljs-string">"- When a skill drives external API writes, assume rate limits: prefer fewer larger writes, avoid tight one-item loops, serialize bursts when possible, and respect 429/Retry-After."</span>,
    trimmed,
    <span class="hljs-string">""</span>,
  ];
}

<span class="hljs-keyword">function</span> buildMemorySection(params: {
  isMinimal: boolean;
  availableTools: Set&lt;string&gt;;
  citationsMode?: MemoryCitationsMode;
}) {
  <span class="hljs-keyword">if</span> (params.isMinimal) {
    <span class="hljs-built_in">return</span> [];
  }
  <span class="hljs-keyword">if</span> (!params.availableTools.has(<span class="hljs-string">"memory_search"</span>) &amp;&amp; !params.availableTools.has(<span class="hljs-string">"memory_get"</span>)) {
    <span class="hljs-built_in">return</span> [];
  }
  const lines = [
    <span class="hljs-string">"## Memory Recall"</span>,
    <span class="hljs-string">"Before answering anything about prior work, decisions, dates, people, preferences, or todos: run memory_search on MEMORY.md + memory/*.md; then use memory_get to pull only the needed lines. If low confidence after search, say you checked."</span>,
  ];
  <span class="hljs-keyword">if</span> (params.citationsMode === <span class="hljs-string">"off"</span>) {
    lines.push(
      <span class="hljs-string">"Citations are disabled: do not mention file paths or line numbers in replies unless the user explicitly asks."</span>,
    );
  } <span class="hljs-keyword">else</span> {
    lines.push(
      <span class="hljs-string">"Citations: include Source: &lt;path#line&gt; when it helps the user verify memory snippets."</span>,
    );
  }
  lines.push(<span class="hljs-string">""</span>);
  <span class="hljs-built_in">return</span> lines;
}

工具描述：
 label: <span class="hljs-string">"Memory Search"</span>,
    name: <span class="hljs-string">"memory_search"</span>,
    description:
      <span class="hljs-string">"Mandatory recall step: semantically search MEMORY.md + memory/*.md (and optional session transcripts) before answering questions about prior work, decisions, dates, people, preferences, or todos; returns top snippets with path + lines. If response has disabled=true, memory retrieval is unavailable and should be surfaced to the user."</span>,
    parameters: MemorySearchSchema,
</code></pre>
<p data-tool="mdnice编辑器">有三点：</p>
<ol data-tool="mdnice编辑器">
<li>
<section><strong>触发条件写得具体</strong>：prior work / decisions / dates / people / preferences / todos。模型不需要猜「算不算记忆相关」。</section>
</li>
<li>
<section><strong>两阶段召回</strong>：先 <code>memory_search</code> 找片段，再 <code>memory_get</code> 精读少量行，控制注入体积。</section>
</li>
<li>
<section><strong>引用策略可控</strong>：<code>citationsMode</code> 可以关掉，避免模型动不动把路径行号甩出来（对产品形态很重要）。</section>
</li>
</ol>
<p data-tool="mdnice编辑器">两阶段召回比「一次性把相关文件 wholefile 塞进去」靠谱太多。</p>
<h1 data-tool="mdnice编辑器"><span class="content">7. 召回结果怎么进上下文</span></h1>
<p data-tool="mdnice编辑器">toolResult 消息是关键通道</p>
<p data-tool="mdnice编辑器">很多人以为「记忆」是把内容写进 system prompt。OpenClaw 不是这么干的。</p>
<p data-tool="mdnice编辑器">召回结果会以工具执行结果的形式进入会话消息列表，后续模型调用自然「看得到」。</p>
<ul data-tool="mdnice编辑器">
<li>
<section>工具返回会被包装成 JSON 文本块（参考内容定位 <code>jsonResult()</code> 在 <code>src/agents/tools/common.ts:230-239</code>）。</section>
</li>
<li>
<section>tool 返回会被标准化成 <code>content[] + details</code>（参考内容定位 <code>src/agents/pi-tool-definition-adapter.ts</code> 的 normalize）。</section>
</li>
<li>
<section>这些 toolResult 会被追加到 session messages，下一次模型调用会携带。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">这条通道对排障非常友好：我们可以在 transcript 里看到「这次回答之前它到底召回了什么」。而且 toolResult 天然可控预算、可控格式，比让模型把记忆揉进自由文本稳得多。</p>
<h1 data-tool="mdnice编辑器"><span class="content">8. 写入时机</span></h1>
<h2 data-tool="mdnice编辑器"><span class="content">8.1 会话快照写入</span></h2>
<p data-tool="mdnice编辑器">人为触发的「切会话」</p>
<p data-tool="mdnice编辑器">当执行 <code>/new</code>、<code>/reset</code> 时，上一段会话尾部会被抽取成 <code>YYYY-MM-DD-&lt;slug&gt;.md</code>。这是「防断片」写入，价值是保住最近上下文。</p>
<p data-tool="mdnice编辑器">坑：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>抽取的 N 条对话里可能包含敏感信息。它会落在 <code>memory/</code>，并进入索引。</section>
</li>
<li>
<section>如果你把 workspace 目录同步到团队共享盘或提交到 repo，泄露面会扩大。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">我们的做法：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>明确区分「个人 workspace」和「团队 workspace」。个人的 <code>memory/</code> 默认不进 repo。</section>
</li>
<li>
<section>开启 citations 时，产品侧要想清楚是否允许暴露路径与行号。</section>
</li>
</ul>
<h2 data-tool="mdnice编辑器"><span class="content">8.2 短期记忆写入</span></h2>
<p data-tool="mdnice编辑器">「需要我们调教，告诉她哪些重要」。</p>
<p data-tool="mdnice编辑器">短期记忆要走「稀疏高密度」路线：条目少，但每条都能在未来的某个问题上直接复用。写入策略要围绕「将来会搜什么」来定，不要围绕「当下发生了什么」来记流水账。</p>
<h2 data-tool="mdnice编辑器"><span class="content">8.3 长期记忆更新</span></h2>
<p data-tool="mdnice编辑器">心跳 / AGENTS.md / cron</p>
<p data-tool="mdnice编辑器">三种更新机制：心跳、核心流程、cron。</p>
<p data-tool="mdnice编辑器">读最近几天短期记忆 → 选值得长期记住的 → 提炼写入 <code>MEMORY.md</code>。</p>
<p data-tool="mdnice编辑器">观点：</p>
<ul data-tool="mdnice编辑器">
<li>
<section><strong>cron 最稳</strong>，可观测、可控、可回滚。</section>
</li>
<li>
<section>心跳更新很容易在负载高时抖动，或者在你最不想更新的时候更新。</section>
</li>
<li>
<section>把它塞进核心流程（AGENTS.md）要谨慎，一旦每次任务都触发提炼，会把延迟拉上去。</section>
</li>
</ul>
<p data-tool="mdnice编辑器">一般做法：</p>
<ul data-tool="mdnice编辑器">
<li>
<section>工作日每天固定一次 consolidation（cron）。</section>
</li>
<li>
<section>遇到重大决策或事故复盘，当天手动提炼写入 <code>MEMORY.md</code>，不等自动化。</section>
</li>
</ul>
<h1 data-tool="mdnice编辑器"><span class="content">9. 怎么「调教」</span></h1>
<p data-tool="mdnice编辑器">把记忆当成协议，不当成日记</p>
<p data-tool="mdnice编辑器">「告诉她哪些重要」。把「重要」拆成几类，每类有明确写入规则，避免模型自由发挥。</p>
<p data-tool="mdnice编辑器">一般的规则：</p>
<ol data-tool="mdnice编辑器">
<li>
<section><strong>稳定偏好</strong>：例如输出格式偏好、技术栈偏好、代码风格偏好。</section>
</li>
<li>
<section><strong>组织事实</strong>：团队结构、系统边界、核心服务依赖、环境约束。</section>
</li>
<li>
<section><strong>关键决策</strong>：ADR 级别的决策，包含时间点与理由。</section>
</li>
<li>
<section><strong>长期目标与在途事项</strong>：能跨周追踪的，不写「今天要做的」。</section>
</li>
<li>
<section><strong>事故教训</strong>：明确到「哪个坑踩过」「如何避免」。</section>
</li>
</ol>
<p data-tool="mdnice编辑器">短期记忆（<code>memory/YYYY-MM-DD.md</code>）会允许更多过程性信息，但要满足一个条件：<strong>未来能被搜索问题命中</strong>。比如你写「今天讨论了 A」，基本没用；你写「决定 A 的原因是 B，后续若出现 C 用 D 回滚」，会有用一些。</p>
<p data-tool="mdnice编辑器">如果希望 <code>memory_search</code> 在关键时刻召回到正确内容，就得用「未来的查询语句」来写记忆。</p>
<p data-tool="mdnice编辑器">以上。</p>
</section>
]]></content:encoded>
			<wfw:commentRss>https://www.phppan.com/2026/03/openclaw-memory/feed/</wfw:commentRss>
		<slash:comments>0</slash:comments>
		</item>
	</channel>
</rss>
