SpringAI学习笔记(三)会话记忆功能

文章目录

SpringAI学习笔记(三)会话记忆功能

三、会话记忆功能

大模型本身是不具备记忆能力的,要想让大模型记住之前聊天的内容,唯一的办法就是把之前聊天的内容与最新的提问一起发给大模型。这里我们就要用到发给大模型的第三个参数 assistant 了。

实现会话记忆功能需要三个必要元素:

1、定义会话存储方式: 将历史会话存储到某个地方,可以是数据库,内存等等。

2、配置会话记忆Advisor: 让Spring帮我们做大模型的会话记忆环绕增强。

3、添加会话id: 每个会话记忆的唯一id,可以是用户的id加用户打开的每个会话框的id。

下面,我们使用 SpringAI 框架自带的 MessageWindowChatMemory 做基本的演示,搭配 SpringAI 框架自带的内存会话管理仓库 InMemoryChatMemoryRepository 类。如果想用Redis、MongoDB、MySQL 等方式进行会话记忆存储跟获取的话,也可以手动实现 ChatMemory 接口 或者 ChatMemoryRepository 接口 来完成。

ChatMemory 接口定义如下

java 复制代码
public interface ChatMemory {
    String DEFAULT_CONVERSATION_ID = "default";
    String CONVERSATION_ID = "chat_memory_conversation_id";

    default void add(String conversationId, Message message) {
        Assert.hasText(conversationId, "conversationId cannot be null or empty");
        Assert.notNull(message, "message cannot be null");
        this.add(conversationId, List.of(message));
    }

    void add(String conversationId, List<Message> messages);

    List<Message> get(String conversationId);

    void clear(String conversationId);
}

从上面接口可以看到,三个接口其实也都是通过会话id的形式来管理会话记忆的。

ChatMemoryRepository 接口定义如下

java 复制代码
public interface ChatMemoryRepository {
    List<String> findConversationIds();

    List<Message> findByConversationId(String conversationId);

    void saveAll(String conversationId, List<Message> messages);

    void deleteByConversationId(String conversationId);
}

步骤一、定义会话存储方式

这里我们以SpringAI 框架自带的 MessageWindowChatMemory 搭配 InMemoryChatMemoryRepository 为例,在配置的Java文件中定义我们的 ChatMemory。

java 复制代码
    /**
     *  注入我们会话记忆管理仓库(使用内存做会话记忆)
     */
    @Bean
    public ChatMemoryRepository chatMemoryRepository(){
        return new InMemoryChatMemoryRepository();
    }
    
    /**
     * 
     * @param chatMemoryRepository  会话记忆仓库,上面我们已经注册了,使用内存做会话记忆
     * @return
     */
    @Bean
    public ChatMemory chatMemory(ChatMemoryRepository chatMemoryRepository) {
        return MessageWindowChatMemory.builder()
                .chatMemoryRepository(chatMemoryRepository)
                .maxMessages(30)    //自定义消息窗口大小
                .build() ;
    }

步骤二、配置会话记忆Advisor

java 复制代码
    //将我们本地模型通过配置直接交给Spring容器做管理,方便使用
    @Bean
    public ChatClient chatClient2Qwen(OpenAiChatModel chatModel, ChatMemory chatMemory){
        return ChatClient.builder(chatModel)
                .defaultSystem("你是一个傻白甜属性的AI助手,你的名字叫小七,请以小七的口吻来回答用户的问题")     //系统设定
                .defaultAdvisors(
                        new SimpleLoggerAdvisor(),                              //配置日志Advisor
                        MessageChatMemoryAdvisor.builder(chatMemory).build()    //配置会话记忆Advisor
                        )             
                .build();
    }

步骤三、添加会话id

java 复制代码
@RestController
@RequestMapping("/ai")
public class ChatController {

    @Resource
    private ChatClient chatClient;

    //流式响应
    @RequestMapping("/streamChat")
    public Flux<String> streamChat(String request, String chatId) {
        return chatClient.prompt()
                .user(request)
                .advisors(advisorSpec -> advisorSpec.param(ChatMemory.CONVERSATION_ID, chatId))
                .stream()
                .content();
    }

}

代码演示

假设我们定义会话的id为用户的id+会话窗口的编号,现在用户的id为1001,打开的会话窗口的编号为A,那么我们的chatId为1001A,我们提问:"假如有10个苹果,怎么平均分给5个人",千问给出的答案如下:

之后,我们使用同一个会话id,继续提问:"如果是分给3个人呢"

可以看到,这里千问是可以根据我们给出的上一个问题的信息做回答的。

但是如果我们现在修改chatId,假设还是用户1001,他打开了会话框B,此时,我们提问:"如果是分给4个人呢",千问就摸不着头脑了:

相关推荐
码途漫谈8 小时前
Easy-Vibe开发篇阅读笔记(二)——前端开发之Figma与MasterGo入门
人工智能·笔记·ai·开源·ai编程·figma
Mr_sst8 小时前
infra-ai模块宏观设计解析:业务与模型之间的中间层核心架构
大数据·人工智能·ai·llama
笨蛋©9 小时前
[实战] 数字化质量管理中的检验计划提效指南:从手工气泡图到AI自动识别
ai·数字化·cad·质量管理·制造业
LaLaLa_OvO9 小时前
jetbrains 的 datagrip 导出csv,中文乱码
笔记
大囚长9 小时前
权力的哲学洞察与反思
笔记
y = xⁿ9 小时前
Redis八股学习日记:布隆过滤器
数据库·redis·学习
imbackneverdie9 小时前
sci期刊示意图、流程图、机制图怎么画?
人工智能·ai·aigc·科研绘图·ai工具·科研工具·ai生图
直奔標竿9 小时前
SpringAI + RAG + MCP + Agent 零基础全栈实战(完结篇)| 27课完整汇总,Java开发者AI转型必看
java·开发语言·人工智能·spring boot·后端·spring
云烟成雨TD9 小时前
Spring AI 1.x 系列【31】向量数据库:进阶使用指南
java·人工智能·spring
CoderJia程序员甲9 小时前
GitHub 热榜项目 - 日榜(2026-04-29)
ai·大模型·github·ai教程