Spring AI - Redis缓存对话

先看效果

对话过程被缓存到了Redis 中。

原理

在上一节我们快速入门了SpringAI,具体文章请查看:快速入门Spring AI

创建 ChatClient 的代码如下:

java 复制代码
this.chatClient = ChatClient.builder(chatModel)
        .defaultSystem(DEFAULT_PROMPT)
        .defaultAdvisors(new MessageChatMemoryAdvisor(new InMemoryChatMemory()))
        .defaultAdvisors(new SimpleLoggerAdvisor())
        .defaultOptions(OpenAiChatOptions.builder().temperature(0.0d).build())
        .build();

其中new MessageChatMemoryAdvisor(new InMemoryChatMemory()) 会将对话缓存在内存中,查看类InMemoryChatMemory 的源码发现,它实际上实现了ChatMemory 接口,实现了 addget以及clear三个方法。

实现

先添加 Redis 的依赖:

xml 复制代码
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

然后定义一个类 RedisChatMemory 实现 ChatMemory 接口,实现三个方法:

java 复制代码
@Override
public void add(String conversationId, List<Message> messages) {
    long time = System.currentTimeMillis();

    for (Message message : messages) {
        redisTemplate.opsForHash().put(conversationId, String.valueOf(time), message);
    }
}

@Override
public List<Message> get(String conversationId, int lastN) {
    Map<Object, Object> entries = redisTemplate.opsForHash().entries(conversationId);
    return entries.entrySet().stream()
            .sorted((o1, o2) -> {
                long time1 = Long.parseLong(o1.getKey().toString());
                long time2 = Long.parseLong(o2.getKey().toString());
                return Long.compare(time1, time2);
            }).limit(lastN)
            .map(e -> new UserMessage(e.getValue().toString()))
            .collect(Collectors.toList());
}

@Override
public void clear(String conversationId) {
    redisTemplate.delete(conversationId);
}

再把 RedisChatMemory 注册成 Bean 对象:

java 复制代码
@Bean
public RedisChatMemory redisChatMemory(RedisTemplate<String, Object> redisTemplate) {
    return new RedisChatMemory(redisTemplate);
}

最后替换 ChatClient 定义中的 InMemoryChatMemory

java 复制代码
this.chatClient = ChatClient.builder(chatModel)
        .defaultSystem(DEFAULT_PROMPT)
//      .defaultAdvisors(new MessageChatMemoryAdvisor(new InMemoryChatMemory()))
        .defaultAdvisors(new MessageChatMemoryAdvisor(redisChatMemory))
        .defaultAdvisors(new SimpleLoggerAdvisor())
        .defaultOptions(OpenAiChatOptions.builder().temperature(0.0d).build())
        .build();

具体代码:代码地址

相关推荐
架构师沉默1 小时前
设计多租户 SaaS 系统,如何做到数据隔离 & 资源配额?
java·后端·架构
Java中文社群3 小时前
重要:Java25正式发布(长期支持版)!
java·后端·面试
每天进步一点_JL4 小时前
JVM 类加载:双亲委派机制
java·后端
用户298698530144 小时前
Java HTML 转 Word 完整指南
java·后端
渣哥4 小时前
原来公平锁和非公平锁差别这么大
java
渣哥5 小时前
99% 的人没搞懂:Semaphore 到底是干啥的?
java
J2K5 小时前
JDK都25了,你还没用过ZGC?那真得补补课了
java·jvm·后端
kfyty7255 小时前
不依赖第三方,不销毁重建,loveqq 框架如何原生实现动态线程池?
java·架构
isysc16 小时前
面了一个校招生,竟然说我是老古董
java·后端·面试
道可到9 小时前
Java 反射现代实践速查表(JDK 11+/17+)
java