Skip to main content
本教程展示如何使用渐进式披露——一种上下文管理技术,其中代理按需加载信息而非预先加载——来实现技能(基于提示的专门指令)。代理通过工具调用加载技能,而不是动态更改系统提示,仅发现并加载每个任务所需的技能。 用例: 想象构建一个代理,帮助在大型企业中跨不同业务垂直领域编写 SQL 查询。您的组织可能为每个垂直领域拥有单独的数据存储,或一个包含数千张表的单一整体数据库。无论哪种情况,预先加载所有模式都会使上下文窗口不堪重负。渐进式披露通过仅在需要时加载相关模式来解决此问题。此架构还使不同的产品负责人和利益相关者能够独立贡献和维护其特定业务垂直领域的技能。 您将构建什么: 一个具有两种技能(销售分析和库存管理)的 SQL 查询助手。代理在其系统提示中看到轻量级技能描述,然后仅在与用户查询相关时通过工具调用加载完整的数据库模式和业务逻辑。
有关具有查询执行、错误纠正和验证功能的 SQL 代理的完整示例,请参阅我们的 SQL 代理教程。本教程重点介绍渐进式披露模式,该模式可应用于任何领域。
渐进式披露由 Anthropic 推广,作为构建可扩展代理技能系统的技术。此方法使用三级架构(元数据 → 核心内容 → 详细资源),代理仅在需要时加载信息。有关此技术的更多信息,请参阅 使用代理技能为代理配备现实世界

工作原理

当用户请求 SQL 查询时,流程如下: 为什么使用渐进式披露:
  • 减少上下文使用 - 仅加载任务所需的 2-3 个技能,而非所有可用技能
  • 实现团队自主性 - 不同团队可以独立开发专门技能(类似于其他多代理架构)
  • 高效扩展 - 添加数十或数百个技能而不会使上下文不堪重负
  • 简化对话历史 - 具有单个对话线程的单一代理
什么是技能: 技能(如 Claude Code 所推广)主要是基于提示的:针对特定业务任务的自包含专门指令单元。在 Claude Code 中,技能作为目录和文件系统上的文件公开,通过文件操作发现。技能通过提示指导行为,并可以提供有关工具使用的信息或包含供编码代理执行的示例代码。
具有渐进式披露的技能可视为 RAG(检索增强生成) 的一种形式,其中每个技能都是一个检索单元——尽管不一定由嵌入或关键词搜索支持,而是由浏览内容的工具(如文件操作或在本教程中直接查找)支持。
权衡:
  • 延迟:按需加载技能需要额外的工具调用,这会增加首次需要每个技能的请求的延迟
  • 工作流控制:基本实现依赖于提示来指导技能使用——您无法在没有自定义逻辑的情况下强制执行硬性约束,例如“始终在技能 B 之前尝试技能 A”
实现您自己的技能系统在构建您自己的技能实现时(如本教程中所做),核心概念是渐进式披露——按需加载信息。除此之外,您在实现方面拥有完全的灵活性:
  • 存储:数据库、S3、内存数据结构或任何后端
  • 发现:直接查找(本教程)、用于大型技能集合的 RAG、文件系统扫描或 API 调用
  • 加载逻辑:自定义延迟特性并添加逻辑以搜索技能内容或对相关性进行排名
  • 副作用:定义技能加载时发生的情况,例如公开与该技能关联的工具(在第 8 节中介绍)
此灵活性使您可以根据性能、存储和工作流控制的具体要求进行优化。

设置

安装

本教程需要 langchain 包:
npm install langchain
有关更多详细信息,请参阅我们的 安装指南

LangSmith

设置 LangSmith 以检查代理内部发生的情况。然后设置以下环境变量:
export LANGSMITH_TRACING="true"
export LANGSMITH_API_KEY="..."

选择 LLM

从 LangChain 的集成套件中选择一个聊天模型:
👉 阅读 OpenAI 聊天模型集成文档
npm install @langchain/openai
import { initChatModel } from "langchain";

process.env.OPENAI_API_KEY = "your-api-key";

const model = await initChatModel("gpt-5.2");

1. 定义技能

首先,定义技能的结构。每个技能都有一个名称、一个简短描述(显示在系统提示中)和完整内容(按需加载):
import { z } from "zod";

// 一个可以逐步披露给代理的技能
const SkillSchema = z.object({
  name: z.string(), // 技能的唯一标识符
  description: z.string(), // 1-2 句描述,显示在系统提示中
  content: z.string(), // 包含详细说明的完整技能内容
});

type Skill = z.infer<typeof SkillSchema>;
现在为 SQL 查询助手定义示例技能。这些技能设计为描述轻量级(预先显示给代理),但内容详细(仅在需要时加载):
import { context } from "langchain";

const SKILLS: Skill[] = [
  {
    name: "sales_analytics",
    description:
      "用于销售数据分析的数据库模式和业务逻辑,包括客户、订单和收入。",
    content: context`
    # 销售分析模式

    ## 表

    ### customers
    - customer_id (主键)
    - name
    - email
    - signup_date
    - status (active/inactive)
    - customer_tier (bronze/silver/gold/platinum)

    ### orders
    - order_id (主键)
    - customer_id (外键 -> customers)
    - order_date
    - status (pending/completed/cancelled/refunded)
    - total_amount
    - sales_region (north/south/east/west)

    ### order_items
    - item_id (主键)
    - order_id (外键 -> orders)
    - product_id
    - quantity
    - unit_price
    - discount_percent

    ## 业务逻辑

    **活跃客户**:
    status = 'active' AND signup_date <= CURRENT_DATE - INTERVAL '90 days'

    **收入计算**:
    仅计算状态为 'completed' 的订单。
    使用 orders 表中的 total_amount,该字段已考虑折扣。

    **客户终身价值 (CLV)**:
    客户所有已完成订单金额的总和。

    **高价值订单**:
    total_amount > 1000 的订单

    ## 示例查询

    -- 获取上一季度按收入排名的前 10 名客户
    SELECT
        c.customer_id,
        c.name,
        c.customer_tier,
        SUM(o.total_amount) as total_revenue
    FROM customers c
    JOIN orders o ON c.customer_id = o.customer_id
    WHERE o.status = 'completed'
    AND o.order_date >= CURRENT_DATE - INTERVAL '3 months'
    GROUP BY c.customer_id, c.name, c.customer_tier
    ORDER BY total_revenue DESC
    LIMIT 10;`,
  },
  {
    name: "inventory_management",
    description:
      "用于库存跟踪的数据库模式和业务逻辑,包括产品、仓库和库存水平。",
    content: context`
    # 库存管理模式

    ## 表

    ### products
    - product_id (主键)
    - product_name
    - sku
    - category
    - unit_cost
    - reorder_point (重新订购前的最低库存水平)
    - discontinued (布尔值)

    ### warehouses
    - warehouse_id (主键)
    - warehouse_name
    - location
    - capacity

    ### inventory
    - inventory_id (主键)
    - product_id (外键 -> products)
    - warehouse_id (外键 -> warehouses)
    - quantity_on_hand
    - last_updated

    ### stock_movements
    - movement_id (主键)
    - product_id (外键 -> products)
    - warehouse_id (外键 -> warehouses)
    - movement_type (inbound/outbound/transfer/adjustment)
    - quantity (正数表示入库,负数表示出库)
    - movement_date
    - reference_number

    ## 业务逻辑

    **可用库存**:
    inventory 表中 quantity_on_hand > 0 的记录

    **需要重新订购的产品**:
    所有仓库中总 quantity_on_hand 小于或等于产品 reorder_point 的产品

    **仅限活跃产品**:
    排除 discontinued = true 的产品,除非专门分析已停产的项目

    **库存估值**:
    每个产品的 quantity_on_hand * unit_cost

    ## 示例查询

    -- 查找所有仓库中低于重新订购点的产品
    SELECT
        p.product_id,
        p.product_name,
        p.reorder_point,
        SUM(i.quantity_on_hand) as total_stock,
        p.unit_cost,
        (p.reorder_point - SUM(i.quantity_on_hand)) as units_to_reorder
    FROM products p
    JOIN inventory i ON p.product_id = i.product_id
    WHERE p.discontinued = false
    GROUP BY p.product_id, p.product_name, p.reorder_point, p.unit_cost
    HAVING SUM(i.quantity_on_hand) <= p.reorder_point
    ORDER BY units_to_reorder DESC;`,
  },
];

2. 创建技能加载工具

创建一个工具,用于按需加载完整的技能内容:
import { tool } from "langchain";
import { z } from "zod";

const loadSkill = tool(
  async ({ skillName }) => {
    // 查找并返回请求的技能
    const skill = SKILLS.find((s) => s.name === skillName);
    if (skill) {
      return `Loaded skill: ${skillName}\n\n${skill.content}`;
    }

    // 未找到技能
    const available = SKILLS.map((s) => s.name).join(", ");
    return `Skill '${skillName}' not found. Available skills: ${available}`;
  },
  {
    name: "load_skill",
    description: `将技能的完整内容加载到代理的上下文中。

当您需要有关如何处理特定类型请求的详细信息时使用此工具。这将为您提供全面的说明、政策和指南。`,
    schema: z.object({
      skillName: z.string().describe("要加载的技能名称"),
    }),
  },
);
load_skill 工具将完整的技能内容作为字符串返回,该字符串作为 ToolMessage 成为对话的一部分。有关创建和使用工具的更多详细信息,请参阅 工具指南

3. 构建技能中间件

创建自定义中间件,将技能描述注入系统提示。此中间件使技能可发现,而无需预先加载其完整内容。
本指南演示如何创建自定义中间件。有关中间件概念和模式的综合指南,请参阅 自定义中间件文档
import { createMiddleware } from "langchain";

// 从 SKILLS 列表构建技能提示
const skillsPrompt = SKILLS.map(
  (skill) => `- **${skill.name}**: ${skill.description}`,
).join("\n");

const skillMiddleware = createMiddleware({
  name: "skillMiddleware",
  tools: [loadSkill],
  wrapModelCall: async (request, handler) => {
    // 构建技能附录
    const skillsAddendum =
      `\n\n## 可用技能\n\n${skillsPrompt}\n\n` +
      "当您需要有关处理特定类型请求的详细信息时,使用 load_skill 工具。";

    // 附加到系统提示
    const newSystemPrompt = request.systemPrompt + skillsAddendum;

    return handler({
      ...request,
      systemPrompt: newSystemPrompt,
    });
  },
});
中间件将技能描述附加到系统提示,使代理了解可用技能,而无需加载其完整内容。load_skill 工具注册为类变量,使其对代理可用。
生产考虑:本教程为简单起见,在 __init__ 中加载技能列表。在生产系统中,您可能希望在 before_agent 钩子中加载技能,以便定期刷新以反映最新更改(例如,添加新技能或修改现有技能)。有关详细信息,请参阅 before_agent 钩子文档

4. 创建支持技能的代理

现在创建具有技能中间件和用于状态持久化的检查点的代理:
import { createAgent } from "langchain";
import { MemorySaver } from "@langchain/langgraph";

// 创建支持技能的代理
const agent = createAgent({
  model,
  systemPrompt: "您是一个 SQL 查询助手,帮助用户 " + "针对业务数据库编写查询。",
  middleware: [skillMiddleware],
  checkpointer: new MemorySaver(),
});
代理现在可以在其系统提示中访问技能描述,并可以在需要时调用 load_skill 来检索完整的技能内容。检查点维护跨轮次的对话历史。

5. 测试渐进式披露

使用需要特定技能知识的问题测试代理:
import { v4 as uuidv4 } from "uuid";

// 此对话线程的配置
const threadId = uuidv4();
const config = { configurable: { thread_id: threadId } };

// 请求 SQL 查询
const result = await agent.invoke(
  {
    messages: [
      {
        role: "user",
        content:
          "编写一个 SQL 查询,查找上个月订单金额超过 1000 美元的所有客户",
      },
    ],
  },
  config,
);

// 打印对话
for (const message of result.messages) {
  console.log(`${message._getType()}: ${message.content}`);
}
预期输出:
================================ Human Message =================================

编写一个 SQL 查询,查找上个月订单金额超过 1000 美元的所有客户
================================== Ai Message ==================================
Tool Calls:
  load_skill (call_abc123)
 Call ID: call_abc123
  Args:
    skill_name: sales_analytics
================================= Tool Message =================================
Name: load_skill

Loaded skill: sales_analytics

# 销售分析模式

## 表

### customers
- customer_id (主键)
- name
- email
- signup_date
- status (active/inactive)
- customer_tier (bronze/silver/gold/platinum)

### orders
- order_id (主键)
- customer_id (外键 -> customers)
- order_date
- status (pending/completed/cancelled/refunded)
- total_amount
- sales_region (north/south/east/west)

[... 模式其余部分 ...]

## 业务逻辑

**高价值订单**:total_amount > 1000 的订单
**收入计算**:仅计算状态为 'completed' 的订单

================================== Ai Message ==================================

这是一个 SQL 查询,用于查找上个月订单金额超过 1000 美元的所有客户:

```sql
SELECT DISTINCT
    c.customer_id,
    c.name,
    c.email,
    c.customer_tier
FROM customers c
JOIN orders o ON c.customer_id = o.customer_id
WHERE o.total_amount > 1000
  AND o.status = 'completed'
  AND o.order_date >= CURRENT_DATE - INTERVAL '1 month'
ORDER BY c.customer_id;
此查询:
  • 将客户与其订单连接
  • 使用 total_amount 字段过滤高价值订单(>1000 美元)
  • 仅包括已完成订单(根据业务逻辑)
  • 限制为上个月的订单
  • 返回不同的客户,以避免同一客户多次符合条件的订单时出现重复
代理在其系统提示中看到轻量级技能描述,识别出问题需要销售数据库知识,调用 load_skill("sales_analytics") 获取完整的模式和业务逻辑,然后使用该信息编写遵循数据库约定的正确查询。

6. 高级:使用自定义状态添加约束

您可以添加约束,以强制某些工具仅在特定技能加载后才可用。这需要在自定义代理状态中跟踪哪些技能已加载。

定义自定义状态

首先,扩展代理状态以跟踪已加载的技能:
import { StateSchema } from "@langchain/langgraph";
import { z } from "zod";

const CustomState = new StateSchema({
  skillsLoaded: z.array(z.string()).optional(), // 跟踪已加载的技能
});

更新 load_skill 以修改状态

修改 load_skill 工具,在技能加载时更新状态:
import { tool, ToolMessage, type ToolRuntime } from "langchain";
import { Command } from "@langchain/langgraph";
import { z } from "zod";

const loadSkill = tool(
  async ({ skillName }, runtime: ToolRuntime<typeof CustomState.State>) => {
    // 查找并返回请求的技能
    const skill = SKILLS.find((s) => s.name === skillName);

    if (skill) {
      const skillContent = `Loaded skill: ${skillName}\n\n${skill.content}`;

      // 更新状态以跟踪已加载的技能
      return new Command({
        update: {
          messages: [
            new ToolMessage({
              content: skillContent,
              tool_call_id: runtime.toolCallId,
            }),
          ],
          skillsLoaded: [skillName],
        },
      });
    }

    // 未找到技能
    const available = SKILLS.map((s) => s.name).join(", ");
    return new Command({
      update: {
        messages: [
          new ToolMessage({
            content: `Skill '${skillName}' not found. Available skills: ${available}`,
            tool_call_id: runtime.toolCallId,
          }),
        ],
      },
    });
  },
  {
    name: "load_skill",
    description: `将技能的完整内容加载到代理的上下文中。`,
    schema: z.object({
      skillName: z.string().describe("要加载的技能名称"),
    }),
  },
);

创建受约束的工具

创建一个仅在特定技能加载后才可用的工具:
const writeSqlQuery = tool(
  async (
    { query, vertical },
    runtime: ToolRuntime<typeof CustomState.State>,
  ) => {
    // 检查是否已加载所需技能
    const skillsLoaded = runtime.state.skillsLoaded ?? [];

    if (!skillsLoaded.includes(vertical)) {
      return (
        `错误:您必须先加载 '${vertical}' 技能 ` +
        `以了解数据库模式,然后才能编写查询。 ` +
        `使用 load_skill('${vertical}') 加载模式。`
      );
    }

    // 验证并格式化查询
    return (
      `SQL 查询,适用于 ${vertical}\n\n` +
      `\`\`\`sql\n${query}\n\`\`\`\n\n` +
      `✓ 查询已根据 ${vertical} 模式验证\n` +
      `准备针对数据库执行。`
    );
  },
  {
    name: "write_sql_query",
    description: `为特定业务垂直领域编写和验证 SQL 查询。

此工具帮助格式化和验证 SQL 查询。您必须先加载适当的技能以了解数据库模式。`,
    schema: z.object({
      query: z.string().describe("要编写的 SQL 查询"),
      vertical: z
        .string()
        .describe("业务垂直领域(sales_analytics 或 inventory_management)"),
    }),
  },
);

更新中间件和代理

更新中间件以使用自定义状态模式:
const skillMiddleware = createMiddleware({
  name: "skillMiddleware",
  stateSchema: CustomState,
  tools: [loadSkill, writeSqlQuery],
  // ... 中间件实现的其余部分保持不变
});
创建具有注册受约束工具的中间件的代理:
const agent = createAgent({
  model,
  systemPrompt: "您是一个 SQL 查询助手,帮助用户 " + "针对业务数据库编写查询。",
  middleware: [skillMiddleware],
  checkpointer: new MemorySaver(),
});
现在,如果代理在加载所需技能之前尝试使用 write_sql_query,它将收到错误消息,提示它先加载适当的技能(例如 sales_analyticsinventory_management)。这确保代理在尝试验证查询之前拥有必要的模式知识。

完整示例

以下是结合本教程所有部分的完整、可运行实现:
import {
  tool,
  createAgent,
  createMiddleware,
  ToolMessage,
  context,
  type ToolRuntime,
} from "langchain";
import { MemorySaver, Command } from "@langchain/langgraph";
import { ChatOpenAI } from "@langchain/openai";
import { v4 as uuidv4 } from "uuid";
import { z } from "zod";

// 一个可以逐步披露给代理的技能
const SkillSchema = z.object({
  name: z.string(), // 技能的唯一标识符
  description: z.string(), // 1-2 句描述,显示在系统提示中
  content: z.string(), // 包含详细说明的完整技能内容
});

type Skill = z.infer<typeof SkillSchema>;

const SKILLS: Skill[] = [
  {
    name: "sales_analytics",
    description:
      "用于销售数据分析的数据库模式和业务逻辑,包括客户、订单和收入。",
    content: context`
    # 销售分析模式

    ## 表

    ### customers
    - customer_id (主键)
    - name
    - email
    - signup_date
    - status (active/inactive)
    - customer_tier (bronze/silver/gold/platinum)

    ### orders
    - order_id (主键)
    - customer_id (外键 -> customers)
    - order_date
    - status (pending/completed/cancelled/refunded)
    - total_amount
    - sales_region (north/south/east/west)

    ### order_items
    - item_id (主键)
    - order_id (外键 -> orders)
    - product_id
    - quantity
    - unit_price
    - discount_percent

    ## 业务逻辑

    **活跃客户**:status = 'active' AND signup_date <= CURRENT_DATE - INTERVAL '90 days'

    **收入计算**:
    仅计算状态为 'completed' 的订单。使用 orders 表中的 total_amount,
    该字段已考虑折扣。

    **客户终身价值 (CLV)**:
    客户所有已完成订单金额的总和。

    **高价值订单**:
    total_amount > 1000 的订单

    ## 示例查询
    -- 获取上一季度按收入排名的前 10 名客户
    SELECT
        c.customer_id,
        c.name,
        c.customer_tier,
        SUM(o.total_amount) as total_revenue
    FROM customers c
    JOIN orders o ON c.customer_id = o.customer_id
    WHERE o.status = 'completed'
    AND o.order_date >= CURRENT_DATE - INTERVAL '3 months'
    GROUP BY c.customer_id, c.name, c.customer_tier
    ORDER BY total_revenue DESC
    LIMIT 10;`,
  },
  {
    name: "inventory_management",
    description:
      "用于库存跟踪的数据库模式和业务逻辑,包括产品、仓库和库存水平。",
    content: context`
    # 库存管理模式

    ## 表

    ### products
    - product_id (主键)
    - product_name
    - sku
    - category
    - unit_cost
    - reorder_point (重新订购前的最低库存水平)
    - discontinued (布尔值)

    ### warehouses
    - warehouse_id (主键)
    - warehouse_name
    - location
    - capacity

    ### inventory
    - inventory_id (主键)
    - product_id (外键 -> products)
    - warehouse_id (外键 -> warehouses)
    - quantity_on_hand
    - last_updated

    ### stock_movements
    - movement_id (主键)
    - product_id (外键 -> products)
    - warehouse_id (外键 -> warehouses)
    - movement_type (inbound/outbound/transfer/adjustment)
    - quantity (正数表示入库,负数表示出库)
    - movement_date
    - reference_number

    ## 业务逻辑

    **可用库存**:
    inventory 表中 quantity_on_hand > 0 的记录

    **需要重新订购的产品**:
    所有仓库中总 quantity_on_hand 小于或等于产品 reorder_point 的产品

    **仅限活跃产品**:
    排除 discontinued = true 的产品,除非专门分析已停产的项目

    **库存估值**:
    每个产品的 quantity_on_hand * unit_cost

    ## 示例查询

    -- 查找所有仓库中低于重新订购点的产品
    SELECT
        p.product_id,
        p.product_name,
        p.reorder_point,
        SUM(i.quantity_on_hand) as total_stock,
        p.unit_cost,
        (p.reorder_point - SUM(i.quantity_on_hand)) as units_to_reorder
    FROM products p
    JOIN inventory i ON p.product_id = i.product_id
    WHERE p.discontinued = false
    GROUP BY p.product_id, p.product_name, p.reorder_point, p.unit_cost
    HAVING SUM(i.quantity_on_hand) <= p.reorder_point
    ORDER BY units_to_reorder DESC;`,
  },
];

// const loadSkill = tool(
//   async ({ skillName }) => {
//     // 查找并返回请求的技能
//     const skill = SKILLS.find((s) => s.name === skillName);
//     if (skill) {
//       return `Loaded skill: ${skillName}\n\n${skill.content}`;
//     }

//     // 未找到技能
//     const available = SKILLS.map((s) => s.name).join(", ");
//     return `Skill '${skillName}' not found. Available skills: ${available}`;
//   },
//   {
//     name: "load_skill",
//     description: `将技能的完整内容加载到代理的上下文中。

// 当您需要有关如何处理特定类型请求的详细信息时使用此工具。这将为您提供全面的说明、政策和指南。`,
//     schema: z.object({
//       skillName: z.string().describe("要加载的技能名称"),
//     }),
//   }
// );

// 从 SKILLS 列表构建技能提示
const skillsPrompt = SKILLS.map(
  (skill) => `- **${skill.name}**: ${skill.description}`,
).join("\n");

const skillMiddleware = createMiddleware({
  name: "skillMiddleware",
  tools: [loadSkill],
  wrapModelCall: async (request, handler) => {
    // 构建技能附录
    const skillsAddendum =
      `\n\n## 可用技能\n\n${skillsPrompt}\n\n` +
      "当您需要有关处理特定类型请求的详细信息时,使用 load_skill 工具。";

    // 附加到系统提示
    const newSystemPrompt = request.systemPrompt + skillsAddendum;

    return handler({
      ...request,
      systemPrompt: newSystemPrompt,
    });
  },
});

const model = new ChatOpenAI({
  model: "gpt-4.1-mini",
  temperature: 0,
});

// 创建支持技能的代理
const agent = createAgent({
  model,
  systemPrompt: "您是一个 SQL 查询助手,帮助用户 " + "针对业务数据库编写查询。",
  middleware: [skillMiddleware],
  checkpointer: new MemorySaver(),
});

// 此对话线程的配置
const threadId = uuidv4();
const config = { configurable: { thread_id: threadId } };

// 请求 SQL 查询
const result = await agent.invoke(
  {
    messages: [
      {
        role: "user",
        content:
          "编写一个 SQL 查询,查找上个月订单金额超过 1000 美元的所有客户",
      },
    ],
  },
  config,
);

// 打印对话
for (const message of result.messages) {
  console.log(`${message.type}: ${message.content}`);
}
此完整示例包括:
  • 具有完整数据库模式的技能定义
  • 用于按需加载的 load_skill 工具
  • 将技能描述注入系统提示的 SkillMiddleware
  • 具有中间件和检查点的代理创建
  • 显示代理如何加载技能和编写 SQL 查询的示例用法
要运行此示例,您需要:
  1. 安装所需包:pip install langchain langchain-openai langgraph
  2. 设置您的 API 密钥(例如 export OPENAI_API_KEY=...
  3. 将模型初始化替换为您首选的 LLM 提供商

实现变体

本教程将技能实现为通过工具调用加载的内存 Python 字典。然而,有几种方法可以使用技能实现渐进式披露:存储后端:
  • 内存中(本教程):技能定义为 Python 数据结构,访问速度快,无 I/O 开销
  • 文件系统(Claude Code 方法):技能作为目录和文件,通过文件操作(如 read_file)发现
  • 远程存储:技能存储在 S3、数据库、Notion 或 API 中,按需获取
技能发现(代理如何了解存在哪些技能):
  • 系统提示列出:技能描述在系统提示中(本教程中使用)
  • 基于文件:通过扫描目录发现技能(Claude Code 方法)
  • 基于注册表:查询技能注册表服务或 API 以获取可用技能
  • 动态查找:通过工具调用列出可用技能
渐进式披露策略(如何加载技能内容):
  • 单次加载:在一个工具调用中加载整个技能内容(本教程中使用)
  • 分页:对于大型技能,分多个页面/块加载技能内容
  • 基于搜索:在特定技能内容中搜索相关部分(例如,使用 grep/read 操作对技能文件进行操作)
  • 分层:首先加载技能概述,然后深入特定子部分
大小考虑(未校准的心理模型 - 针对您的系统进行优化):
  • 小型技能(< 1K 令牌 / ~750 词):可以直接包含在系统提示中,并通过提示缓存进行缓存以节省成本并加快响应速度
  • 中型技能(1-10K 令牌 / ~750-7.5K 词):受益于按需加载以避免上下文开销(本教程)
  • 大型技能(> 10K 令牌 / ~7.5K 词,或 > 上下文窗口的 5-10%):应使用渐进式披露技术,如分页、基于搜索的加载或分层探索,以避免消耗过多上下文
选择取决于您的要求:内存中最快,但需要重新部署以更新技能,而基于文件或远程存储支持动态技能管理,无需更改代码。

渐进式披露与上下文工程

渐进式披露本质上是**上下文工程**技术——您管理代理可用的信息及其可用时间。本教程重点介绍加载数据库模式,但相同原则适用于其他类型的上下文。

结合少样本提示

对于 SQL 查询用例,您可以扩展渐进式披露以动态加载与用户查询匹配的少样本示例示例方法:
  1. 用户询问:“查找 6 个月内未订购的客户”
  2. 代理加载 sales_analytics 模式(如本教程所示)
  3. 代理还通过语义搜索或基于标签的查找加载 2-3 个相关示例查询:
    • 用于查找不活跃客户的查询
    • 具有基于日期过滤的查询
    • 连接客户和订单表的查询
  4. 代理使用模式知识和示例模式编写查询
这种渐进式披露(按需加载模式)和动态少样本提示(加载相关示例)的结合创建了一种强大的上下文工程模式,可扩展到大型知识库,同时提供高质量、有根据的输出。

后续步骤