Spring AI文档学习:多聊天模型配置

前置:application.yml(阿里云百炼作为 OpenAI 兼容端点)

yaml 复制代码
spring:
  ai:
    openai:
      api-key: ${DASHSCOPE_API_KEY}
      base-url: https://dashscope.aliyuncs.com/compatible-mode/v1
      chat:
        options:
          model: qwen-plus

说明:api-key 使用百炼控制台里的 API Key;model 用百炼里在「OpenAI 兼容」下支持的模型名(如 qwen-plusqwen-max 等,以控制台为准)。

方法一:同一模型类型(都是 OpenAiChatModel / 百炼),多个 ChatClient

差别只在 ChatClient 层默认行为(例如不同系统提示),底层仍是同一个注入的 OpenAiChatModel

kotlin 复制代码
import org.springframework.ai.chat.client.ChatClient;
import org.springframework.ai.openai.OpenAiChatModel;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class BailianMultiClientConfig {

    @Bean
    public ChatClient conciseBailianClient(OpenAiChatModel chatModel) {
        return ChatClient.builder(chatModel)
                .defaultSystem("你是简洁助手,回答尽量短。")
                .build();
    }

    @Bean
    public ChatClient detailedBailianClient(OpenAiChatModel chatModel) {
        return ChatClient.builder(chatModel)
                .defaultSystem("你是详细助手,先列要点再展开。")
                .build();
    }
}

@Qualifier("conciseBailianClient") / @Qualifier("detailedBailianClient") 注入即可。

方法二:不同模型类型,多个 ChatClient(百炼 OpenAI 兼容 + 另一家)

这里 第一类仍是百炼(OpenAiChatModel + 上面 yml)。第二类换厂商时用另一种 ChatModel(示例用 Anthropic;需引入对应 Spring AI starter 并配置密钥)。

kotlin 复制代码
import org.springframework.ai.anthropic.AnthropicChatModel;
import org.springframework.ai.chat.client.ChatClient;
import org.springframework.ai.openai.OpenAiChatModel;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

@Configuration
public class MultiVendorChatClientConfig {

    /** 百炼:OpenAI 兼容,底层是 OpenAiChatModel */
    @Bean
    public ChatClient bailianChatClient(OpenAiChatModel openAiChatModel) {
        return ChatClient.builder(openAiChatModel)
                .defaultSystem("你通过阿里云百炼(OpenAI 兼容)回答问题。")
                .build();
    }

    /** 另一家:原生 Anthropic API */
    @Bean
    public ChatClient anthropicChatClient(AnthropicChatModel anthropicChatModel) {
        return ChatClient.builder(anthropicChatModel)
                .defaultSystem("你通过 Anthropic API 回答问题。")
                .build();
    }
}

要点:百炼走 OpenAI 兼容 → OpenAiChatModel;Anthropic 走自己的 AnthropicChatModel,这是「不同模型类型」在 Spring AI 里的典型拆法。

方法三:多个 OpenAI 兼容端点(百炼 + 另一个 OpenAI 兼容服务)

仍只用 OpenAiApi / OpenAiChatModel,通过 mutate()baseUrlapiKey、默认 model。第二个端点可以是:官方 OpenAI、Groq、其它国内兼容网关等,协议形状与 OpenAI 一致即可。

scss 复制代码
import org.springframework.ai.chat.client.ChatClient;
import org.springframework.ai.openai.OpenAiChatModel;
import org.springframework.ai.openai.api.OpenAiApi;
import org.springframework.ai.openai.OpenAiChatOptions;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;

@Service
public class MultiOpenAiCompatibleEndpointsService {

    @Autowired
    private OpenAiChatModel baseChatModel;

    @Autowired
    private OpenAiApi baseOpenAiApi;

    public void compareTwoEndpoints(String userMessage) {
        // 端点 A:阿里云百炼(北京 OpenAI 兼容)
        OpenAiApi bailianApi = baseOpenAiApi.mutate()
                .baseUrl("https://dashscope.aliyuncs.com/compatible-mode/v1")
                .apiKey(System.getenv("DASHSCOPE_API_KEY"))
                .build();

        OpenAiChatModel bailianModel = baseChatModel.mutate()
                .openAiApi(bailianApi)
                .defaultOptions(OpenAiChatOptions.builder()
                        .model("qwen-plus")
                        .temperature(0.3)
                        .build())
                .build();

        // 端点 B:另一个 OpenAI 兼容服务(示例:请换成你真实的 URL / Key / 模型名)
        OpenAiApi otherApi = baseOpenAiApi.mutate()
                .baseUrl("https://api.openai.com/v1")  // 或其它兼容网关
                .apiKey(System.getenv("OPENAI_API_KEY"))
                .build();

        OpenAiChatModel otherModel = baseChatModel.mutate()
                .openAiApi(otherApi)
                .defaultOptions(OpenAiChatOptions.builder()
                        .model("gpt-4o-mini")
                        .temperature(0.3)
                        .build())
                .build();

        String r1 = ChatClient.builder(bailianModel).build()
                .prompt(userMessage).call().content();
        String r2 = ChatClient.builder(otherModel).build()
                .prompt(userMessage).call().content();

        // 使用 r1 / r2
    }
}

若两个都是百炼,只是 地域或模型不同,把「端点 B」改成例如新加坡兼容地址,并只改 model 即可

相关推荐
ltl15 小时前
位置编码:为什么需要它,为什么用正弦
后端
明月_清风15 小时前
Go 函数设计的工程智慧:多返回值、闭包与那些"反直觉"的选择
后端·go
却尘15 小时前
一个 `&` 引发的血案:改完配置 pipeline 装聋作哑,顺便重学了 Python/Go/Java
后端·go
倚栏听风雨15 小时前
Spring AI 实战:用 JdbcChatMemory + MySQL 给 AI 接上「长期记忆」
后端
我叫黑大帅16 小时前
最简单的生产-消费者,你都会遇到哪些问题?
后端·面试·go
ze^016 小时前
Day01 Web应用&架构搭建&域名源码&站库分离&MVC模型&解析受限&对应路径
安全·web安全·架构·mvc·安全架构
刀法如飞17 小时前
Palantir Ontology 数据结构分析,与ER/OOP/DDD有什么区别?
人工智能·算法·架构
swipe17 小时前
Agentic RAG:用 LangGraph 构建会路由、会纠错、会收敛的闭环 RAG
后端·langchain·llm
狼与自由17 小时前
微服务网关演化
微服务·云原生·架构
折哥的程序人生 · 物流技术专研17 小时前
《Java 100 天进阶之路》第23篇:缓冲区数据结构 ByteBuffer
java·开发语言·数据结构·后端·面试·求职招聘