SpringBoot中集成LangChain4j实现集成阿里百炼平台进行AI对话记忆功能和对话隔离功能

场景

SpringBoot中集成LangChain4j实现集成阿里百炼平台进行AI快速对话:

https://blog.csdn.net/BADAO_LIUMANG_QIZHI/article/details/160214877

在如上快速实现AI对话后,但是对话没有记忆功能,历史对话无法记住。

且不同用户之间的对话没有隔离。

在 LangChain4j 中,要实现多轮对话的记忆和用户间的会话隔离,主要是通过 @MemoryId 参数来区分的。

LangChain4j的 AiServices

LangChain4j的 AiServices 能通过注解声明式地构建带有记忆功能、系统提示等高级能力的AI服务。

使用示例:

定义服务接口:使用 @AiService 注解定义一个接口,并通过 @SystemMessage 设置系统提示词。

复制代码
import dev.langchain4j.service.AiService;
import dev.langchain4j.service.SystemMessage;

@AiService
public interface Assistant {

    @SystemMessage("你是一位知识渊博的AI助手,请用中文友好地回答用户的问题。")
    String chat(String userMessage);
}

注入并使用服务:LangChain4j会自动创建该接口的代理实现,你可以像使用普通Spring Bean一样注入它。

复制代码
@RestController
public class AssistantController {

    private final Assistant assistant;

    public AssistantController(Assistant assistant) {
        this.assistant = assistant;
    }

    @GetMapping("/ai/assistant")
    public String assistantChat(@RequestParam(value = "message") String message) {
        return assistant.chat(message);
    }
}

对话记忆 (ChatMemory):

为了支持多轮连续对话,可以集成 ChatMemory 组件。MessageWindowChatMemory 是一个内存实现,

适合开发测试

@MemoryId 与会话隔离

LangChain4j 的 AiServices 支持一种非常优雅的会话隔离方式。

你只需要在AI服务接口的方法参数中,增加一个带有 @MemoryId 注解的参数,

框架就会自动为你管理不同ID对应的独立对话记忆。

注:

博客:
https://blog.csdn.net/badao_liumang_qizhi

实现

1. 改造 AI 服务接口

修改之前定义的 Assistant 接口,为 chat 方法增加 @MemoryId 参数和 @UserMessage 注解,

这样可以更精确地控制传给LLM的用户消息内容。

复制代码
import dev.langchain4j.service.MemoryId;
import dev.langchain4j.service.SystemMessage;
import dev.langchain4j.service.UserMessage;
import dev.langchain4j.service.spring.AiService;

@AiService
public interface Assistant {

    @SystemMessage("你是一位知识渊博的AI助手,请用中文友好地回答用户的问题。")
    String chat(@MemoryId Long memoryId, @UserMessage String userMessage);

}

这里的关键是 @MemoryId Long memoryId。

当你用不同的 memoryId(比如用户ID)调用这个方法时,LangChain4j 会自动为每个ID维护一个独立的聊天记忆窗口

2、配置 ChatMemoryProvider Bean

创建一个配置类,定义一个 ChatMemoryProvider 的 Bean。

这个 Bean 的作用是告诉 AiServices,当遇到一个新的 memoryId 时,应该如何创建对应的 ChatMemory 实例。

复制代码
import dev.langchain4j.memory.chat.ChatMemoryProvider;
import dev.langchain4j.memory.chat.MessageWindowChatMemory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;


@Configuration
public class AiConfig {

    // 定义一个Bean,用于为每个memoryId提供独立的ChatMemory
    @Bean
    public ChatMemoryProvider chatMemoryProvider() {
        // MessageWindowChatMemory 是一个基于消息数量的滑动窗口[reference:4]
        // 这里设置最多保留最近的10条消息,超出后会从历史记录中移除最早的消息[reference:5][reference:6]
        return memoryId -> MessageWindowChatMemory.withMaxMessages(10);
    }
}

这段代码为每个 memoryId 创建了一个最多能记住 10 条 最新消息的 MessageWindowChatMemory 实例

3. 在 Controller 中调用

最后,在 Controller 里注入刚才的 Assistant Bean,

并在处理请求时从某个地方(比如HTTP请求头)获取并传入用户的唯一标识作为 memoryId。

复制代码
import com.badao.ai.service.Assistant;
import org.springframework.web.bind.annotation.*;

@RestController
public class AssistantController {

    private final Assistant assistant;

    public AssistantController(Assistant assistant) {
        this.assistant = assistant;
    }

    @GetMapping("/ai/assistant")
    public String chat(@RequestHeader("X-User-Id") Long userId,
                       @RequestParam(value = "message") String message) {
        // 将用户ID作为 memoryId 传递给AI服务
        return assistant.chat(userId, message);
    }
}

通过 HTTP 请求头 X-User-Id 来获取用户的唯一标识,并将其作为 memoryId 传入。

4、关键依赖版本与注意事项

依赖版本:

需要确保你的 pom.xml 中 LangChain4j 的版本是 1.0.0-beta3 或更高,

因为之前的版本在 @UserMessage 注解的支持上可能存在差异。你之前使用的 1.0.0-beta1 可能无法正常工作,建议进行升级。

记忆存储:

MessageWindowChatMemory 的对话记录是存储在内存中的,一旦应用重启就会丢失。

如果需要持久化保存,可以实现 ChatMemoryStore 接口,将数据存入数据库(如MongoDB)或Redis等外部存储中。

生产环境:

在生产环境中,建议使用更精确的 TokenWindowChatMemory,它基于Token数量进行淘汰,能更好地控制成本

附示例中的pom文件:

复制代码
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>3.2.5</version>
    </parent>

    <groupId>com.example</groupId>
    <artifactId>spring-langchain4j-bailian</artifactId>
    <version>1.0</version>

    <properties>
        <java.version>17</java.version>
        <langchain4j.version>1.0.0-beta3</langchain4j.version>
    </properties>

    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>dev.langchain4j</groupId>
                <artifactId>langchain4j-community-bom</artifactId>
                <version>${langchain4j.version}</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>

    <dependencies>
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <!-- LangChain4j 核心 Spring Boot Starter -->
        <dependency>
            <groupId>dev.langchain4j</groupId>
            <artifactId>langchain4j-spring-boot-starter</artifactId>
             <version>${langchain4j.version}</version>
        </dependency>

        <!-- 阿里百炼 DashScope 集成 Starter -->
        <dependency>
            <groupId>dev.langchain4j</groupId>
            <artifactId>langchain4j-community-dashscope-spring-boot-starter</artifactId>
        </dependency>

        <!-- 显式添加 DashScope 核心依赖(Starter 有时不会传递) -->
        <dependency>
            <groupId>dev.langchain4j</groupId>
            <artifactId>langchain4j-community-dashscope</artifactId>
        </dependency>
    </dependencies>

5、测试

使用API工具(如Postman),模拟两个不同用户的请求进行测试:

模拟用户A(用户ID为 1):

curl -H "X-User-Id: 1" "http://localhost:885/ai/assistant?message=你好,我叫张三。"

模拟用户A(用户ID为 1):

curl -H "X-User-Id: 1" "http://localhost:885/ai/assistant?message=我叫什么名字?"

预期AI助手能够回答出"张三"。

模拟用户B(用户ID为 2):

curl -H "X-User-Id: 2" "http://localhost:885/ai/assistant?message=我叫什么名字?"

预期AI助手因为缺乏上下文,会表示不知道或需要用户告知。

相关推荐
薛定猫AI几秒前
【深度解析】从 Chatbot 到 AI 数字队友:Claude 高阶能力、模型选型与 API 实战
人工智能
熊文豪7 分钟前
从零到一的AI产品演进:用Claude Code与蓝耘MaaS打造多模型聚合聊天工具全纪实
人工智能·microsoft·蓝耘
码途漫谈24 分钟前
Easy-Vibe高级开发篇阅读笔记(五)——CC教程之Agent Teams
人工智能·笔记·ai·开源·ai编程
deephub25 分钟前
为什么 MCP 在协议层会有 prompt injection的问题:工具描述如何劫持 agent 上下文
人工智能·深度学习·大语言模型·ai-agent·mcp
likerhood28 分钟前
Fastjson中的JSON.parseObject()详细讲解
java·json
乔代码嘚29 分钟前
2026 AI大模型全套资料免费领!30天从入门到架构部署,附面试真题与行业报告
人工智能·语言模型·面试·大模型·产品经理·ai大模型·大模型学习
KNeeg_31 分钟前
黑马点评完整代码(RabbitMQ优化)+简历编写+面试重点 ⭐
java·redis·后端·spring·面试·职场和发展·黑马点评
Java后端的Ai之路32 分钟前
以为AI开发就是调接口?一场25K的面试让我看到真相,原来真正的技术深度在这!
人工智能·面试·职场和发展·agent·ai应用开发
塔能物联运维34 分钟前
两相液冷:从“散热”到“控温”,重构高密度算力的热管理系统
大数据·人工智能
lazy熊35 分钟前
AI 编程新手教程(1):用 Codex 读懂一个现成项目
人工智能