Spring AI 从入门到精通-Prompt 工程

5. Prompt 工程:不是玄学,是手艺

5.1 什么是 Prompt?

Prompt 是你发给 AI 的"指令包"。它不只是"一句问题",而是由多个 Message 组成的结构化数据:
#mermaid-svg-bJLZeD2k5U6XIBH7{font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}@keyframes edge-animation-frame{from{stroke-dashoffset:0;}}@keyframes dash{to{stroke-dashoffset:0;}}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-animation-slow{stroke-dasharray:9,5!important;stroke-dashoffset:900;animation:dash 50s linear infinite;stroke-linecap:round;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-animation-fast{stroke-dasharray:9,5!important;stroke-dashoffset:900;animation:dash 20s linear infinite;stroke-linecap:round;}#mermaid-svg-bJLZeD2k5U6XIBH7 .error-icon{fill:#552222;}#mermaid-svg-bJLZeD2k5U6XIBH7 .error-text{fill:#552222;stroke:#552222;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-thickness-normal{stroke-width:1px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-thickness-thick{stroke-width:3.5px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-pattern-solid{stroke-dasharray:0;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-thickness-invisible{stroke-width:0;fill:none;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-pattern-dashed{stroke-dasharray:3;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edge-pattern-dotted{stroke-dasharray:2;}#mermaid-svg-bJLZeD2k5U6XIBH7 .marker{fill:#333333;stroke:#333333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .marker.cross{stroke:#333333;}#mermaid-svg-bJLZeD2k5U6XIBH7 svg{font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;}#mermaid-svg-bJLZeD2k5U6XIBH7 p{margin:0;}#mermaid-svg-bJLZeD2k5U6XIBH7 .label{font-family:"trebuchet ms",verdana,arial,sans-serif;color:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster-label text{fill:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster-label span{color:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster-label span p{background-color:transparent;}#mermaid-svg-bJLZeD2k5U6XIBH7 .label text,#mermaid-svg-bJLZeD2k5U6XIBH7 span{fill:#333;color:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .node rect,#mermaid-svg-bJLZeD2k5U6XIBH7 .node circle,#mermaid-svg-bJLZeD2k5U6XIBH7 .node ellipse,#mermaid-svg-bJLZeD2k5U6XIBH7 .node polygon,#mermaid-svg-bJLZeD2k5U6XIBH7 .node path{fill:#ECECFF;stroke:#9370DB;stroke-width:1px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .rough-node .label text,#mermaid-svg-bJLZeD2k5U6XIBH7 .node .label text,#mermaid-svg-bJLZeD2k5U6XIBH7 .image-shape .label,#mermaid-svg-bJLZeD2k5U6XIBH7 .icon-shape .label{text-anchor:middle;}#mermaid-svg-bJLZeD2k5U6XIBH7 .node .katex path{fill:#000;stroke:#000;stroke-width:1px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .rough-node .label,#mermaid-svg-bJLZeD2k5U6XIBH7 .node .label,#mermaid-svg-bJLZeD2k5U6XIBH7 .image-shape .label,#mermaid-svg-bJLZeD2k5U6XIBH7 .icon-shape .label{text-align:center;}#mermaid-svg-bJLZeD2k5U6XIBH7 .node.clickable{cursor:pointer;}#mermaid-svg-bJLZeD2k5U6XIBH7 .root .anchor path{fill:#333333!important;stroke-width:0;stroke:#333333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .arrowheadPath{fill:#333333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edgePath .path{stroke:#333333;stroke-width:2.0px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .flowchart-link{stroke:#333333;fill:none;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edgeLabel{background-color:rgba(232,232,232, 0.8);text-align:center;}#mermaid-svg-bJLZeD2k5U6XIBH7 .edgeLabel p{background-color:rgba(232,232,232, 0.8);}#mermaid-svg-bJLZeD2k5U6XIBH7 .edgeLabel rect{opacity:0.5;background-color:rgba(232,232,232, 0.8);fill:rgba(232,232,232, 0.8);}#mermaid-svg-bJLZeD2k5U6XIBH7 .labelBkg{background-color:rgba(232, 232, 232, 0.5);}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster rect{fill:#ffffde;stroke:#aaaa33;stroke-width:1px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster text{fill:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 .cluster span{color:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 div.mermaidTooltip{position:absolute;text-align:center;max-width:200px;padding:2px;font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:12px;background:hsl(80, 100%, 96.2745098039%);border:1px solid #aaaa33;border-radius:2px;pointer-events:none;z-index:100;}#mermaid-svg-bJLZeD2k5U6XIBH7 .flowchartTitleText{text-anchor:middle;font-size:18px;fill:#333;}#mermaid-svg-bJLZeD2k5U6XIBH7 rect.text{fill:none;stroke-width:0;}#mermaid-svg-bJLZeD2k5U6XIBH7 .icon-shape,#mermaid-svg-bJLZeD2k5U6XIBH7 .image-shape{background-color:rgba(232,232,232, 0.8);text-align:center;}#mermaid-svg-bJLZeD2k5U6XIBH7 .icon-shape p,#mermaid-svg-bJLZeD2k5U6XIBH7 .image-shape p{background-color:rgba(232,232,232, 0.8);padding:2px;}#mermaid-svg-bJLZeD2k5U6XIBH7 .icon-shape .label rect,#mermaid-svg-bJLZeD2k5U6XIBH7 .image-shape .label rect{opacity:0.5;background-color:rgba(232,232,232, 0.8);fill:rgba(232,232,232, 0.8);}#mermaid-svg-bJLZeD2k5U6XIBH7 .label-icon{display:inline-block;height:1em;overflow:visible;vertical-align:-0.125em;}#mermaid-svg-bJLZeD2k5U6XIBH7 .node .label-icon path{fill:currentColor;stroke:revert;stroke-width:revert;}#mermaid-svg-bJLZeD2k5U6XIBH7 :root{--mermaid-font-family:"trebuchet ms",verdana,arial,sans-serif;} Prompt
SystemMessage

系统指令:你是谁、怎么回答
UserMessage

用户输入:实际问题
AssistantMessage

历史回答:之前的对话
ToolMessage

工具结果:函数调用返回

5.2 SystemMessage:给 AI 定"人设"

java 复制代码
String reply = chatClient.prompt()
        .system("""
            你是一个经验丰富的 Java 面试官。
            规则:
            1. 每次只问一个问题
            2. 根据回答的深度追问
            3. 不要直接给答案,引导思考
            4. 回答用中文
            """)
        .user("我想练习 Java 并发相关的面试题")
        .call()
        .content();

5.3 模板变量:动态 Prompt

java 复制代码
// 基本用法
String reply = chatClient.prompt()
        .user(u -> u.text("列出 {count} 部由 {composer} 配乐的电影")
                .param("count", 5)
                .param("composer", "久石让"))
        .call()
        .content();

// 多个参数
String reply2 = chatClient.prompt()
        .user(u -> u.text("比较 {lang1} 和 {lang2} 在 {aspect} 方面的差异")
                .param("lang1", "Java")
                .param("lang2", "Kotlin")
                .param("aspect", "协程支持"))
        .call()
        .content();

底层原理: ChatClient 内部用 PromptTemplate + StTemplateRenderer(基于 StringTemplate 引擎)做模板替换。默认用 {} 作为占位符,如果你想在 Prompt 里放 JSON,可以换分隔符:

java 复制代码
String reply = chatClient.prompt()
        .user(u -> u.text("翻译下面的话:<text>")
                .param("text", "Hello World"))
        .templateRenderer(StTemplateRenderer.builder()
                .startDelimiterToken('<')
                .endDelimiterToken('>')
                .build())
        .call()
        .content();

5.4 一个 Prompt 模板的实战案例

假设你要做一个"根据用户输入生成 SQL"的功能:

java 复制代码
@Service
public class SqlGenerator {

    private final ChatClient chatClient;

    public SqlGenerator(ChatClient.Builder builder) {
        this.chatClient = builder
                .defaultSystem("""
                    你是一个 SQL 生成器。根据用户的自然语言描述生成 SQL 语句。
                    
                    规则:
                    1. 只输出 SQL,不要解释
                    2. 表名是 users(id, name, email, age, city, created_at)
                    3. 使用标准 SQL 语法
                    4. 默认 LIMIT 100
                    """)
                .defaultOptions(OpenAiChatOptions.builder()
                        .temperature(0.0)  // SQL 需要精确,不要创意
                        .build())
                .build();
    }

    public String generateSql(String naturalLanguage) {
        return chatClient.prompt()
                .user(naturalLanguage)
                .call()
                .content();
    }
}

// 使用
String sql = sqlGenerator.generateSql("找出北京所有年龄大于 25 岁的用户,按注册时间倒序");
// 输出:SELECT * FROM users WHERE city = '北京' AND age > 25 ORDER BY created_at DESC LIMIT 100

5.5 Prompt 设计原则(Head First 精华版)

  1. 说人话,说清楚。 不要模棱两可。"列出 5 个"比"列一些"好。
  2. 给角色。 "你是一个 XX 专家"能让 AI 切换知识库。
  3. 给规则。 明确告诉 AI 什么能做、什么不能做。
  4. 给例子。 Few-shot prompting 是提升质量的最快方式。
  5. 给格式。 如果期望 JSON,就明确说"输出 JSON,不要有其他内容"。
  6. 分步骤。 复杂任务分解成"首先...然后...最后..."。

相关推荐
yaoxin52112317 小时前
434. Java 日期时间 API - Period 基于日期的时间段
java·开发语言·python
何极光18 小时前
IDEA集成Maven
java·maven·intellij-idea
程序员二叉18 小时前
【JUC】ThreadLocal底层原理|内存泄漏|弱引用|跨线程传递方案
java·开发语言·面试·职场和发展·juc
程序员二叉18 小时前
【JUC】线程池全套深度详解|参数|流程|拒绝策略|调优|异常处理
java·开发语言·jvm·算法·面试·juc
老马识途2.018 小时前
在AI的帮助下理解spring的启动过程
java·前端·spring
青山木18 小时前
Hot 100 --- 轮转数组
java·数据结构·算法
Qt程序员19 小时前
掌握 Linux 内核调度:从原理到实现(进程篇)
java·开发语言
code bean19 小时前
【LangChain】检索器完全指南:从向量检索到生产级 RAG 架构
java·开发语言·微服务
大白菜和MySQL19 小时前
java应用排查高线程
java·python
KobeSacre19 小时前
ReentrantLock源码
java