Spring AI 从入门到精通-结构化输出

6. 结构化输出:让 AI 交作业,不是写散文

6.1 问题:AI 输出的是文本,但你要的是对象

AI 默认返回的是自然语言文本。但你的程序需要的是结构化的 Java 对象。怎么办?

java 复制代码
// 你想要的是这个
record Person(String name, int age, String city) {}

// 但 AI 给你的是这个
"张三,今年 25 岁,住在北京。他的朋友李四..."

6.2 BeanOutputConverter:文本 → Java Bean

java 复制代码
// 1. 定义目标类型
record Person(String name, int age, String city) {}

// 2. 用 ChatClient 的 entity() 方法
Person person = chatClient.prompt()
        .user("生成一个随机的人物信息")
        .call()
        .entity(Person.class);

System.out.println(person.name());  // 张三
System.out.println(person.age());   // 28
System.out.println(person.city());  // 上海

发生了什么? Spring AI 在后台做了两件事:

  1. 根据 Person 类生成 JSON Schema,追加到 Prompt 后面作为格式指令
  2. 拿到 AI 返回的 JSON 后,用 Jackson 反序列化成 Person 对象

6.3 泛型类型:返回 List

java 复制代码
// 返回 List<Person>
List<Person> people = chatClient.prompt()
        .user("生成 5 个随机的人物信息")
        .call()
        .entity(new ParameterizedTypeReference<List<Person>>() {});

people.forEach(p -> System.out.println(p.name() + ", " + p.age()));

💡 ParameterizedTypeReference 是什么? 因为 Java 在运行时会擦除泛型类型(List<Person> 在运行时只是 List),Spring AI 需要通过这个匿名内部类的方式拿到实际的泛型参数类型。记住这个写法模板就行,每次需要返回泛型集合时照抄。

6.4 MapOutputConverter:返回 Map

java 复制代码
Map<String, Object> result = chatClient.prompt()
        .user(u -> u.text("列出 {subject} 的 {count} 个属性,用 JSON 格式")
                .param("subject", "一个 Java 开发者的简历")
                .param("count", 5))
        .call()
        .entity(new ParameterizedTypeReference<Map<String, Object>>() {});

System.out.println(result);
// {name=张三, experience=5年, skills=[Java, Spring, MySQL, Redis], ...}

6.5 ListOutputConverter:返回列表

java 复制代码
List<String> flavors = chatClient.prompt()
        .user(u -> u.text("列出 {count} 种 {subject}")
                .param("count", 10)
                .param("subject", "冰淇淋口味"))
        .call()
        .entity(new ListOutputConverter(new DefaultConversionService()));

flavors.forEach(System.out::println);
// 巧克力
// 香草
// 草莓
// ...

6.6 Native Structured Output:更可靠的结构化

一些模型(GPT-4o、Claude 3.5 等)原生支持结构化输出。Spring AI 用 Advisor 参数来启用:

java 复制代码
Person person = chatClient.prompt()
        .advisors(AdvisorParams.ENABLE_NATIVE_STRUCTURED_OUTPUT)
        .user("生成一个随机的人物信息")
        .call()
        .entity(Person.class);

区别: 普通模式是在 Prompt 里加"请输出 JSON",AI 可能不听话。Native 模式是把 JSON Schema 传给模型的原生 API,模型保证输出符合 Schema。

全局启用:

java 复制代码
@Bean
ChatClient chatClient(ChatClient.Builder builder) {
    return builder
            .defaultAdvisors(AdvisorParams.ENABLE_NATIVE_STRUCTURED_OUTPUT)
            .build();
}

6.7 底层用法:手动使用 BeanOutputConverter

有时候你需要手动控制转换过程:

java 复制代码
BeanOutputConverter<Person> converter = new BeanOutputConverter<>(Person.class);
String format = converter.getFormat();  // 拿到格式指令

String userTemplate = """
    生成一个随机人物信息。
    {format}
    """;

Prompt prompt = new Prompt(
    PromptTemplate.builder()
        .template(userTemplate)
        .variables(Map.of("format", format))
        .build()
        .createMessage()
);

Generation generation = chatModel.call(prompt).getResult();
Person person = converter.convert(generation.getOutput().getText());

💡 什么时候用底层? 当你需要自定义 Prompt 模板结构,不想用 ChatClient 的自动拼接时。


相关推荐
米小虾7 小时前
Loop Engineering —— 循环的设计与自主执行
人工智能·agent
ZhengEnCi7 小时前
J7A-高级Java工程师面试三道灵魂拷问-深度广度与工程素养的终极检验
java·后端
米小虾7 小时前
Harness Engineering —— 系统的安全护栏
人工智能·agent
火山引擎开发者社区8 小时前
积分当钱花,火山引擎开发者激励计划首月消费双倍回馈
人工智能
aqi008 小时前
15天学会AI应用开发(十)把文本嵌入模型换成国产模型
人工智能·python·ai编程
MobotStone9 小时前
为什么在AI时代,“好奇心”成了最值钱的能力?
人工智能
武子康10 小时前
调查研究-200 llama.cpp b9754:一次很小但很关键的 Agent 工具调用修复
人工智能·agent·llama
Ralph_Salar10 小时前
从0到1搭建AI智能支付风控助手Stage1-RAG知识库升级 — 元数据让检索更精准
人工智能