claude code 工程化学习1:记忆系统 CLAUDE.md

0 阅读13分钟

CLAUDE.md 是什么?

和AI 协作,不知道你有没有这样的经历。

第一次对话:
你:帮我写一个用户登录接口
Claude:好的,这是一个基础的登录接口...(使用 Express + JavaScript)

你:我们项目用的是 FastifyTypeScript
Claude:好的,让我重新写...

第二次对话:
你:帮我写一个订单创建接口
Claude:好的,这是一个基础的订单接口...(又用 Express + JavaScript)

你:(崩溃)我们用 FastifyTypeScript

在刚刚开始使用 Claude Code 时,这种情况常见。对于小项目,我多说几次需求,倒也无所谓。

但是时间长了,项目逐渐复杂的时候,如果每次新对话,Claude 都让我从零开始,如果它不记得你的项目用什么技术栈、什么代码风格、什么团队规范——那这种“失忆症”让人抓狂。

CLAUDE.md 就是治疗这种失忆症的药。

它是一份给 Claude 的“项目入职手册”——Claude 每次开始对话时,都会自动阅读这份手册,了解你的项目背景,明确它在干活时应该遵循的一系列底层规则。

当你在项目目录启动 Claude Code 时,发生的“记忆系统初始化”过程如下图所示。

image.png

就像你给新员工一份入职手册,他读完之后就知道公司的规矩。不同的是,Claude 每次对话都会重新“入职”——所以这份手册必须简洁有效。

Claude code 五层记忆

Claude code 支持五个层级的记忆,就像洋葱一样,从外到内,按层级结构组织——高层级的文件优先加载,为底层文件提供基础。

image.png

如果是个人或小团队,可以直接跳过企业级设定这一层,不影响任何使用。

用户级内容设定

用户级内容设定承载的是你的全局偏好,即跨所有项目生效的个人偏好,如个人代码风格,沟通语言设置,通用工作习惯等。比如说我希望所有的 PPT 都是 16:9,黑体字。这种设置就应该放在此处。

位置:~/.claude/CLAUDE.md

示例:

# 个人偏好

## 沟通方式
- 使用中文回复
- 代码注释使用英文
- 解释简洁直接,不要过多铺垫

## 通用代码风格
- 缩进使用 2 空格
- 优先使用 async/await
- 变量命名使用 camelCase
- 常量命名使用 UPPER_SNAKE_CASE

## 我的常用工具
- 包管理器: uv
- 编辑器: VS Code
- 终端: zsh

用户户级记忆会被项目级覆盖。如果你个人喜欢 2 空格缩进,但项目要求 4 空格,那就用 4 空格。

项目级团队共享规范

团队共享规范是团队共享的项目知识,应该提交到 Git。适合存放的内容包括项目架构和技术栈、团队编码规范、重要的设计决策和常用命令。

位置:项目根目录的  ./CLAUDE.md

示例(一个后端 API 项目):

# 项目:订单服务 API

## 技术栈
- Node.js 20 + TypeScript
- FastifyWeb 框架)
- PrismaORM)
- PostgreSQL + Redis
- Zod(数据验证)

## 目录结构
src/ 
├── routes/ # 路由定义 
├── controllers/ # 请求处理 
├── services/ # 业务逻辑 
├── repositories/ # 数据访问 
├── schemas/ # Zod schemas 
└── types/ # 类型定义

## API 响应格式
```typescript
interface ApiResponse<T> {
  success: boolean;
  data?: T;
  error?: { code: string; message: string };
}
编码规范
- TypeScript strict 模式
- 禁止使用 any,使用 unknown + 类型守卫
- 所有 API 端点必须有 Zod schema 验证
- 业务错误使用自定义 Error 类
常用命令
- pnpm dev - 启动开发服务器
- pnpm test - 运行测试
- pnpm prisma migrate dev - 运行数据库迁移

本地个人工作空间

个人工作空间用于记载个人工作笔记,不提交到 Git,适合内容包括本地环境配置、个人调试技巧、当前工作备注,敏感信息(测试账号等)。

位置:项目根目录的./CLAUDE.local.md

示例如下:

# 本地开发笔记

## 我的环境
- 本地 API: http://localhost:3000
- 测试数据库: order_service_dev
- Redis: localhost:6379

## 测试账号
- admin@test.com / test123
- user@test.com / test123

## 当前工作
- 正在重构支付模块
- 参考 PR #234 的讨论
- 周五前完成

## 调试技巧
- 订单状态机日志: LOG_LEVEL=debug pnpm dev
- 查看 Redis 缓存: redis-cli KEYS "order:*"

记得把  CLAUDE.local.md  加入  .gitignore!

当在项目越来越大,周期越来越长的时候,一个属于自己的本地记忆空间其实还蛮有用的。我在和 Claude Code 多轮对话之后,Claude 也会自动压缩对话历史。经过一系列提示词之后,我自己也不知道自己进行到哪一步了,想查一下以前的提示词,或者几天前和 Claude Code 的关键讨论,但是无处寻踪了。

而拥有一个记忆空间,定期把关键内容更新就能够解决这个问题(自己更新或者让 Claude 帮忙更新关键点都行)。

规则目录 rules

这是一个比较高阶的技巧。Rules 是按主题组织的规则文件,支持条件作用域(也就是视情况来确定是否加载该记忆内容),适合场景包括 CLAUDE.md 变得太长时,不同文件类型需要不同规范时,以及前后端分离的项目。

位置:.claude/rules/*.md

目录结构:

.claude/
└── rules/
    ├── typescript.md      # TypeScript 规范
    ├── testing.md         # 测试规范
    ├── api-design.md      # API 设计规范
    └── security.md        # 安全规范

条件作用域示例:.claude/rules/testing.md

---
paths:
  - "src/**/*.test.ts"
  - "tests/**/*.ts"
---

# 测试规范

## 命名
- 单元测试: `*.test.ts`
- 集成测试: `*.integration.test.ts`

## 结构
使用 Arrange-Act-Assert 模式:

```typescript
describe('OrderService', () => {
  describe('createOrder', () => {
    it('should create order when stock is available', async () => {
      // Arrange
      const mockProduct = createMockProduct({ stock: 10 });

      // Act
      const order = await orderService.createOrder(mockProduct.id, 1);

      // Assert
      expect(order.status).toBe('created');
    });
  });
});

## 覆盖率要求
- 业务逻辑: > 80%
- 工具函数: > 90%
- 路由/控制器: 可以较低

此处的关键特性是paths字段让这个规则只在编辑测试文件时生效,不会浪费其他场景的上下文空间。

如何编写高效的 CLAUDE.md

如果你只能记住一句话,那就是 CLAUDE.md 写得好不好,直接决定了 Claude 是靠谱同事,还是每次都要重新培训的实习生

下面我们就来讨论 CLAUDE.md 编写要遵循的核心原则,了解怎么写,才值得每次都被加载进上下文。

核心原则 1:Less is More

CLAUDE.md 的每一行,都会在每一次对话开始时被自动注入上下文。这意味着一件事:冗余不是无害的,而是持续消耗的。所以保持精简不是建议,而是必须。

核心原则 2:具体优于泛泛

先来看一个非常常见、但几乎没有任何效果的写法。

# 项目规范
## 代码质量
请写出高质量的代码。代码应该是可读的。使用有意义的变量名。
保持代码整洁。遵循最佳实践。不要写重复的代码。

这些话没有一句是错的,但问题在于——Claude 本来就知道这些。它们不会改变 Claude 的任何决策,只会白白占用上下文空间。这些话对人类尚且含糊,对模型来说,更是几乎等于什么都没说。

真正有价值的 CLAUDE.md,应该长这样。

# 项目规范

## TypeScript
- 使用 `interface` 定义对象结构,`type` 用于联合类型
- 禁止 `any`,使用 `unknown` + 类型守卫
- 函数参数 > 3 个时,使用对象参数

## 错误处理
```typescript
// 业务错误
throw new BusinessError('ORDER_NOT_FOUND', '订单不存在');

// 验证错误(Zod 自动抛出)
const data = orderSchema.parse(input);

// controller 中不要 try-catch
// 由全局错误中间件统一处理

两者的差异非常明确,后者不是模糊要求“要高质量”,而是给出了如何做才算高质量;不是“注意错误处理”,而是具体的错误模型;不是抽象描述,而是可直接模仿的代码形态。

这里有个简单的判断标准,如果你不写,Claude 也大概率会做对,那就不要写

核心原则 3:关键三问题 WHY / WHAT / HOW

一份真正“能用”的 CLAUDE.md,通常都在回答三个问题。不是一次性回答,而是在关键地方给出明确指引。

WHY —— 为什么要这样做?

## 为什么使用 Zod?
- TypeScript 只有编译时类型检查
- API 输入需要运行时验证
- Zod 可以同时生成 TS 类型和验证逻辑
- 错误信息自动生成,对用户友好

这一部分的作用,不是让 Claude “记住一个库”,而是让它理解背后的决策逻辑。当 Claude 明白了为什么,它在面对相似但不完全相同的场景时,才更可能做出一致的判断。

WHAT —— 具体要做什么,不要做什么?

## 数据库操作规范
- 所有查询通过 Prisma ORM
- 复杂查询封装在 `src/repositories/`
- 禁止在 controller/service 中直接写 SQL
- 事务使用 `prisma.$transaction()`

这一部分的重点是边界。什么是允许的,什么是禁止的,决策应该发生在哪一层?对 Claude 来说,这比“最佳实践”四个字重要得多。

HOW —— 按什么步骤去做?

## 创建新 API 端点

1.`src/schemas/` 创建请求/响应 Zod schema
2.`src/routes/` 添加路由定义
3.`src/controllers/` 实现请求处理
4.`src/services/` 实现业务逻辑
5.`tests/` 添加测试用例

示例参考: `src/routes/orders.ts`

当步骤清晰、路径明确、还有参考文件时,Claude 才会稳定复用同一套工作流,而不是每次自由发挥

核心原则 4:渐进式披露:不要把一切都塞进 CLAUDE.md

CLAUDE.md 的职责是定义默认决策,而不是承载全部知识。

对于非核心、但可能被用到的内容,正确的做法是引用,而不是复制。

# 项目规范

## 核心
[精简的核心规范]

## 详细文档
- 数据库设计: 见 `docs/database.md`
- API 规范: 见 `docs/api-spec.md`
- 部署流程: 见 `docs/deployment.md`

这样做有两个好处:

  • CLAUDE.md 保持轻量,启动成本低。

  • 当 Claude 需要进一步的细节信息时,可以按需读取引用文件。

CLAUDE.md 实战演练

场景一:为新项目创建记忆

假设你刚接手一个 React + TypeScript 前端项目,让我们从零配置记忆。

Step 1:创建基础 CLAUDE.md

先通过 /init 命令自动初始化 CLAUDE.md 文件,或使用下面的命令在项目根目录手动创建记忆文件。

touch CLAUDE.md

然后创建如下的内容。

# 项目:电商平台前端

## 技术栈
- React 18 + TypeScript
- Vite 构建
- TanStack Query(数据获取)
- Zustand(状态管理)
- Tailwind CSS

## 目录结构
src/ 
├── components/ # 组件 
│ ├── ui/ # 基础 UI 
│ └── features/ # 功能组件 
├── pages/ # 页面 
├── hooks/ # 自定义 Hooks 
├── stores/ # Zustand stores 
├── api/ # API 调用 
└── types/ # 类型定义

## 组件规范
- 函数组件 + Hooks
- Props 接口命名: `XxxProps`
- 一个组件一个目录: `Button/index.tsx`

## 状态管理
- 服务端状态: TanStack Query
- 客户端状态: Zustand
- 本地状态: useState

## 常用命令
- `pnpm dev` - 开发服务器
- `pnpm build` - 构建
- `pnpm test` - 测试

Step 2:创建本地记忆

touch CLAUDE.local.md
echo "CLAUDE.local.md" >> .gitignore

然后创建如下的内容。

# 本地笔记

## 环境
- API: http://localhost:8080
- Mock: 使用 MSW

## 当前任务
- 重构购物车组件
- 截止: 本周五

Step 3:添加条件规则(可选)

mkdir -p .claude/rules

然后创建如下.claude/rules/testing.md:

---
paths:
  - "src/**/*.test.tsx"
  - "src/**/*.test.ts"
---

# 测试规范

- 使用 Vitest + React Testing Library
- 测试文件放在同目录: `Button.test.tsx`
- 优先测试用户行为,而非实现细节

```typescript
// ✅ 好
expect(screen.getByRole('button')).toBeEnabled();

// ❌ 不好
expect(component.state.isLoading).toBe(false);

场景二:优化已有的 CLAUDE.md

假设你的 CLAUDE.md 已经有 500 行,Claude 开始变慢。此时就需要给它瘦个身,做一些优化了。我们可以分三步走。

Step 1:识别核心内容

可以问自己:哪些内容是每次对话都需要的?

下面是对于项目整体的一个规划示例,目的是使 CLAUDE.md 保有一个简单而清晰的结构。

image.png

Step 2:拆分成独立文件

详细的 API 文档、数据库表结构和部署流程虽然重要,但是完全没有必要每次都读入 Claude 内存,可以移动到单独文件,精简原来的 CLAUDE.md 。

## 核心规范
[精简内容]

## 详细参考
- API 端点清单: @docs/api.md
- 数据库 Schema: @prisma/schema.prisma
- 部署配置: @docs/deploy.md

Step 3:使用条件规则

可以考虑进一步把测试规范、前端规范、后端规范拆分到  .claude/rules/,并设置  paths条件。

场景三:记忆管理命令

要查看当前记忆,在 Claude Code 中输入:

/memory

就会显示当前加载的所有记忆内容和来源。

编辑记忆的命令参数如下。

/memory edit         # 编辑项目级 CLAUDE.md
/memory edit user    # 编辑用户级记忆
/memory edit local   # 编辑本地级记忆

你也可以通过自然语言指令,让 Claude 帮你更新记忆!

你:请记住,我们项目使用 pnpm 而不是 npm

Claude:好的,我可以将这个信息添加到项目的 CLAUDE.md 中。
        要我现在更新吗?

Auto Memory 解读

最后的一个重要知识点是,Claude Code 本身拥有自动记忆功能,随着项目的演进和对话的深入,会在 ~/.claude/projects//memory/ 目录下自动生成 Auto Memory,用于记录模型在项目中学习到的模式、调试经验与结构认知。

这意味着,Claude Code 的“记忆”并不是单一文件,而是一种多层叠加的上下文注入架构:有些是人为编写的长期规则,有些是组织级强制策略,还有一些是模型自动沉淀的经验笔记。

CLAUDE.md 决定“系统被告知什么”,而 Auto Memory 决定“系统在实践中学到了什么”。记忆因此成为一种结构化的工程能力,而不是简单的对话缓存。

总结

CLAUDE.md 是有层次的记忆。它的意义,是把项目规范、编码风格和团队约定中反复强调的共识,从对话中抽离出来,变成一次配置、长期生效的默认规则。

然而记忆本身是有成本的。

CLAUDE.md 会在每一次对话开始时自动加载。这意味着它并不适合承载所有信息,而只适合存放每次都必须知道的内容。当记忆过多、层级混乱,Claude 的行为反而会变得迟钝甚至不稳定。

因此,理解加载顺序、控制记忆体量、区分团队规范与个人偏好,并不是简单的进阶技巧,而是能否长期使用这套机制的前提。

当 Claude 响应明显变慢,经常出现上下文长度警告,而且 Claude“忘记”对话早期的内容时,可以采用 “.md 瘦身三步法”:精简 → 拆分 → 条件规则。

image.png

从更高的层面看,这一讲讨论的并不只是一个配置文件,而是一种新的协作方式:把隐性的经验、默认的判断和反复纠正的规则,提前固化成结构

这样,Claude 才能从一个需要不断校准的工具,逐渐变成一个行为稳定、风格一致的协作伙伴。

文章内容来源自极客时间。