当前位置: 首页 > article >正文

你不是金鱼——Spring AI 聊天记忆从“重启即失忆”到 MySQL 持久化的生产级改造实录

你不是金鱼——Spring AI 聊天记忆从“重启即失忆”到 MySQL 持久化的生产级改造实录一、问题不是“记不住”,而是系统根本没有记忆层很多团队第一次做 AI 对话应用时,都会产生一个错觉:模型这么聪明,应该能“记住”我刚刚说过的话现实是:大语言模型是无状态的每次调用都是一次全新的推理所谓“记忆”,本质上是应用层把历史消息重新拼进 Prompt这意味着,一旦你的聊天历史只存在 JVM 内存里,服务重启、Pod 漂移、实例扩缩容、节点故障,都会让用户的上下文瞬间消失。这不是体验问题,而是架构问题。在真实生产环境里,聊天记忆至少要同时满足四件事:重启不丢多实例共享多租户隔离成本和延迟可控如果做不到这四点,所谓“AI 客服”“AI 助手”“AI Copilot”都只是单机 Demo。二、先厘清一个常被混淆的概念:Memory 不是 HistorySpring AI 官方文档对这件事的定义非常重要:Chat Memory:提供给模型参与当前推理的上下文记忆Chat History:完整对话流水,用于审计、回放、分析、合规、训练这两个概念在工程上绝不能混为一谈。如果你把完整历史全部塞给模型,会遇到三个问题:Token 成本持续上涨响应延迟不断变大无关历史污染当前意图,回答质量反而下降所以一个成熟系统通常是“双轨存储”:一条轨道给模型用:短期记忆,窗口化、可裁剪、面向推理一条轨道给业务用:完整历史,可审计、可检索、可归档这也是为什么 Spring AI 把记忆抽象做成了两层:策略层和存储层。三、Spring AI 聊天记忆到底是怎么工作的Spring AI 的聊天记忆可以拆成三个核心角色:┌─────────────────────────────────────────────────────────┐ │ ChatClient + Advisor Chain │ │ 负责把“记忆”在请求前注入,在响应后写回 │ └─────────────────────────────────────────────────────────┘ │ ▼ ┌─────────────────────────────────────────────────────────┐ │ ChatMemory │ │ 负责决定“哪些消息应该保留给模型” │ │ 内置实现:MessageWindowChatMemory │ └─────────────────────────────────────────────────────────┘ │ ▼ ┌─────────────────────────────────────────────────────────┐ │ ChatMemoryRepository │ │ 负责决定“消息存在哪里” │ │ InMemory / JDBC / Cassandra / Neo4j / Mongo 等 │ └─────────────────────────────────────────────────────────┘1.ChatMemory是策略,不是数据库ChatMemory的职责不是持久化,而是控制记忆边界。默认实现MessageWindowChatMemory维护一个滑动窗口,默认保留最近 20 条消息,同时保留系统消息。这层解决的是:当前会话应该带多少上下文给模型老消息什么时候被驱逐如何避免上下文无限膨胀2.ChatMemoryRepository是存储抽象ChatMemoryRepository只做一件事:存和取。官方 API 的方法很简单:public interface ChatMemoryRepository { ListString findConversationIds(); ListMessage findByConversationId(String conversationId); void saveAll(String conversationId, ListMessage messages); void deleteByConversationId(String conversationId); }这里有一个很容易被忽略、但对高并发架构非常关键的点:saveAll(...)的语义是“替换当前会话的记忆窗口”它不是逐条 append这意味着在默认模式下,窗口越大,每次写入放大越明显。假设窗口保留 40 条消息,那么一轮对话结束后,Repository 很可能要把这 40 条的当前视图整体保存一次。工程上这会带来两个直接结论:记忆窗口不是越大越好“短期记忆”和“完整归档”最好拆开设计3. Advisor 才是“自动注入记忆”的关键在ChatClient里,真正把记忆串起来的是 Advisor。以MessageChatMemoryAdvisor为例,它在一轮对话里做两件事:请求发给模型之前,根据conversationId取出历史消息并拼入 Prompt模型响应回来之后,把本轮消息更新回ChatMemory可以把它理解成一个“记忆拦截器链”:四、为什么默认InMemoryChatMemoryRepository一定会在生产翻车Spring AI 默认会自动装配:InMemoryChatMemoryRepositoryMessageWindowChatMemory它非常适合本地开发,但不适合生产,原因不是“性能差”,而是“边界错了”。1. 生命周期绑定 JVM内存记忆和进程同生共死:应用重启即丢失滚动发布即丢失OOM 重启即丢失2. 无法跨实例共享在 Kubernetes 或 ECS 多副本部署下:用户第一次命中 Pod A第二次命中 Pod B两个实例各自持有不同内存结果就是“机器人像失忆了一样前后矛盾”。3. 无法治理和审计你看不到:某个租户最近占了多少记忆数据哪些会话增长异常某个回答为什么会引用那段历史所以内存实现的本质定位只有一个:本地验证不是试运行,不是灰度,也不是生产。五、生产选型:为什么第一站通常应该是 MySQL很多人会问,聊天记忆是不是应该直接上 Redis?答案是:未必。先看三种典型选型:方案优点缺点适合场景InMemory零成本、零配置重启丢失、无法共享本地开发JDBC + MySQL强一致、易审计、易运维、SQL 能力强写放大明显、热点会话压力集中大多数生产系统Redis 自定义 Repository低延迟、天然 TTL、横向扩展好审计弱、历史检索弱、持久化语义需要额外设计高频短会话、超高并发场景对大多数企业应用,MySQL 是最稳妥的第一站,原因有四个:团队熟悉,运维成熟强一致语义清晰容易做租户隔离、索引、归档、备份便于和历史表、工

相关文章:

你不是金鱼——Spring AI 聊天记忆从“重启即失忆”到 MySQL 持久化的生产级改造实录

你不是金鱼——Spring AI 聊天记忆从“重启即失忆”到 MySQL 持久化的生产级改造实录 一、问题不是“记不住”,而是系统根本没有记忆层 很多团队第一次做 AI 对话应用时,都会产生一个错觉: 模型这么聪明,应该能“记住”我刚刚说过的话 现实是: 大语言模型是无状态的 每次…...

用Windows Package Manager (winget) 一键搞定.NET全家桶更新:从安装到升级的保姆级指南

用Windows Package Manager (winget) 高效管理.NET开发环境:从安装到版本控制的完整实践 在Windows平台上进行.NET开发的开发者们,是否厌倦了每次更新运行时和SDK时的手动下载、安装过程?随着.NET 8、7、6等多个版本的并行维护,开…...

深入PyTorch源码:torch.nn.utils.clip_grad_norm_是如何计算并裁剪梯度范数的?

深入PyTorch源码:torch.nn.utils.clip_grad_norm_梯度裁剪机制全解析 在深度学习的训练过程中,梯度爆炸是一个常见且棘手的问题。当神经网络的层数加深,参数数量增多时,反向传播过程中梯度可能会呈指数级增长,最终导致…...

EMQX数据备份恢复踩坑实录:从CLI命令到实战避坑指南

EMQX数据备份恢复实战:从版本兼容到集群操作的深度避坑指南 凌晨三点,服务器告警铃声刺破了寂静——EMQX集群升级后的数据迁移失败了。Dashboard上闪烁的红色警告提示着3000多个物联网设备即将失去连接权限。这不是我第一次面对数据备份恢复的烂摊子&…...

从一次网站迁移踩坑说起:手把手教你用308状态码做SEO友好的永久重定向(附工具实测)

网站迁移实战:308状态码如何帮你避开SEO陷阱与数据丢失 去年我们团队负责一个大型电商平台的域名迁移项目,原本以为简单的301重定向就能搞定,结果上线后一周内表单提交量骤降40%,搜索引擎流量也出现明显波动。这次踩坑经历让我深刻…...

BackupPC实战:误删服务器文件后,我是如何用5分钟快速恢复的?

BackupPC实战:误删服务器文件后,我是如何用5分钟快速恢复的? 那天下午3点17分,咖啡杯在桌上留下半圈冷凝水。我正调试新部署的K8s集群时,突然收到开发组的紧急消息:"测试环境的/demo目录被整个删除了&…...

告别UI卡顿!TouchGFX资源外置到QSPI Flash的完整配置指南(含链接脚本详解)

TouchGFX资源外置到QSPI Flash的工程实践与性能优化 在嵌入式UI开发中,TouchGFX凭借其出色的渲染效果和流畅的动画表现,已成为许多高端嵌入式设备的首选框架。然而,随着UI复杂度的提升,图片、字体等资源往往会占用大量存储空间&am…...

Arm CoreLink CI-700 QoS与MPAM寄存器配置详解

1. Arm CoreLink CI-700 QoS与MPAM寄存器配置概述 在复杂的SoC设计中,资源争用和内存访问冲突是影响系统性能的关键瓶颈。Arm CoreLink CI-700作为高性能一致性互连架构,通过硬件级QoS(服务质量)和MPAM(内存分区与监控…...

告别串行等待!用BAP直接访问接口,让你的芯片MBIST测试快人一步

告别串行等待!用BAP直接访问接口让你的芯片MBIST测试快人一步 在芯片测试领域,时间就是金钱。当传统IJTAG串行配置的时钟周期像沙漏般缓慢流逝时,BAP(BIST Access Port)的直接访问接口犹如为存储器测试装上了涡轮引擎。…...

Rockchip平台Camera调试避坑实录:从‘isp no free cp buffer’到‘crc errors’的硬件级排查指南

Rockchip平台Camera硬件调试实战:从信号完整性到寄存器配置的深度解析 当你在Rockchip平台上调试Camera模组时,是否遇到过这样的场景:系统能跑,但log里疯狂报错,图像异常,而软件配置看起来一切正常&#xf…...

蓝桥杯单片机省赛必备:手把手教你用STC15驱动DS18B20(附完整代码与数码管显示)

蓝桥杯单片机竞赛实战:DS18B20温度采集与数码管显示的深度优化 在蓝桥杯单片机竞赛中,DS18B20温度传感器的稳定读取与显示是高频考点。本文将基于STC15单片机,从硬件原理到代码实现,完整解析温度采集系统的构建过程,并…...

命令行集成多模态AI助手:jules工具实战与架构解析

1. 项目概述:当命令行遇上多模态AI助手如果你和我一样,大部分工作时间都泡在终端里,那你肯定对命令行的高效和专注深有体会。但有时候,面对一个复杂的系统问题,或者需要快速理解一段陌生的代码逻辑时,你可能…...

网盘直链下载助手:八大平台一键解析,告别限速烦恼

网盘直链下载助手:八大平台一键解析,告别限速烦恼 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 ,支持 百度网盘 / 阿里云盘 / 中国移动云盘…...

阴阳师自动化脚本:智能任务托管与高效游戏管理解决方案

阴阳师自动化脚本:智能任务托管与高效游戏管理解决方案 【免费下载链接】OnmyojiAutoScript Onmyoji Auto Script | 阴阳师脚本 项目地址: https://gitcode.com/gh_mirrors/on/OnmyojiAutoScript OnmyojiAutoScript(简称OAS)是一款专为…...

别再只会拖拽了!用Vue.draggable + JSON Schema,手把手教你打造企业级低代码组件库

从JSON Schema到企业级低代码平台:Vue.draggable组件库架构实战 在数字化转型浪潮中,企业级低代码平台正成为提升开发效率的关键基础设施。不同于简单的拖拽拼接,真正的低代码平台需要建立完整的组件生态体系,而这一切的基础在于如…...

Android 13 CTS测试一次过:从环境搭建到报告解读的保姆级避坑指南

Android 13 CTS测试全流程实战:从零搭建到精准排错的进阶指南 在Android生态系统中,兼容性测试套件(CTS)是确保设备符合Google规范的关键环节。随着Android 13的发布,测试工具链和标准都发生了显著变化,这对…...

FusionCube管理员密码忘了别慌:深入IAM表结构,揭秘密码重置原理

FusionCube管理员密码重置的底层逻辑解析 当FusionCube系统的管理员密码遗失时,许多工程师的第一反应是寻找操作步骤文档。但真正理解密码重置背后的机制,能让你在更复杂的场景下游刃有余。本文将带你深入IAM模块的数据库结构,揭示密码存储与…...

3步构建Windows任务栏透明化工具TranslucentTB的容器化开发环境

3步构建Windows任务栏透明化工具TranslucentTB的容器化开发环境 【免费下载链接】TranslucentTB A lightweight utility that makes the Windows taskbar translucent/transparent. 项目地址: https://gitcode.com/gh_mirrors/tr/TranslucentTB TranslucentTB是一款轻量…...

AgentMesh:基于文件系统的多AI智能体协同开发协议

1. 项目概述:当你的AI助手们开始“开会”最近在折腾一个多AI智能体协同开发的项目,遇到了一个非常典型且头疼的问题:我同时在用Claude Code重构后端,用Cursor写前端组件,还用着Codex CLI生成测试代码。它们各自在自己的…...

Signal协议的双棘轮算法:为什么WhatsApp和Messenger的聊天记录无法被批量破解?

Signal协议的双棘轮算法:为什么WhatsApp和Messenger的聊天记录无法被批量破解? 想象一下,你和朋友之间的每次对话都使用一本全新的密码本,而每次发送消息后这本密码本就会自动销毁。更神奇的是,即使有人偷走了其中一本…...

3种策略彻底解决TranslucentTB任务栏透明工具在Windows 11更新后的启动问题

3种策略彻底解决TranslucentTB任务栏透明工具在Windows 11更新后的启动问题 【免费下载链接】TranslucentTB A lightweight utility that makes the Windows taskbar translucent/transparent. 项目地址: https://gitcode.com/gh_mirrors/tr/TranslucentTB TranslucentT…...

深入解析Godot文档仓库:从Sphinx构建到社区贡献全流程

1. 从源码到手册:深入拆解 Godot 文档仓库的构建与贡献 如果你正在使用 Godot Engine 开发游戏,那么 godotengine/godot-docs 这个仓库就是你绕不开的“官方百科全书”。它远不止是一个简单的文档网站源码,而是一个由社区驱动、基于 Sphin…...

Ortur激光雕刻机对焦与线距优化实践

1. Ortur Laser Master 2 Pro激光对焦与最佳线距设置实践作为一名长期使用各类激光雕刻设备的创客,我发现很多用户在刚接触Ortur Laser Master 2 Pro(以下简称LM2Pro)时,最常遇到的两个核心问题就是激光对焦不准和雕刻线条间距设置…...

Portwell PCOM-B65A COM Express模块:工业边缘计算新选择

1. Portwell PCOM-B65A COM Express模块概述Portwell最新推出的PCOM-B65A COM Express Type 6 Basic模块,标志着嵌入式系统领域的一次重要升级。这款模块采用了Intel Core Ultra处理器平台(原代号Meteor Lake),为工业自动化和边缘…...

自动驾驶端到端系统OpenREAD:LLM与多模态融合实践

1. 项目背景与核心价值 自动驾驶技术发展到今天,已经从单纯的感知决策分离架构逐步向端到端一体化演进。OpenREAD系统的出现,恰好解决了当前行业面临的几个关键痛点:传统模块化架构的误差累积问题、多传感器数据融合的复杂性、以及面对长尾场…...

从Guava到Redisson:一次搞懂Java中两种布隆过滤器的选型、实战与避坑指南

Guava与Redisson布隆过滤器实战:Java开发者选型决策指南 当系统面临海量数据存在性判断需求时,传统哈希表的内存消耗与查询效率问题便暴露无遗。作为概率型数据结构的经典实现,布隆过滤器以极低的内存占用实现了O(1)时间复杂度的元素存在性检…...

避坑指南:UniApp下载文件到手机本地,你可能遇到的3个平台兼容性问题与解决方案

UniApp跨平台文件下载实战:深度解析三大兼容性陷阱与系统级优化策略 在移动应用开发中,文件下载功能看似基础,却暗藏诸多平台差异的"暗礁"。最近接手的一个企业办公应用项目就遇到了典型问题——在测试阶段表现完美的文档下载功能&…...

Arm处理器性能分析框架与优化实践

1. Arm处理器性能分析基础框架 在SoC设计领域,性能分析从来都不是简单的跑分游戏。当我第一次拿到Arm Cortex-M7芯片时,发现标称的200MHz主频在实际应用中竟然跑不出预期的性能,这个教训让我深刻认识到:真正的性能优化需要建立系统…...

崩坏星穹铁道终极自动化指南:三月七小助手如何每天为你节省2小时?

崩坏星穹铁道终极自动化指南:三月七小助手如何每天为你节省2小时? 【免费下载链接】March7thAssistant 崩坏:星穹铁道全自动 三月七小助手 项目地址: https://gitcode.com/gh_mirrors/ma/March7thAssistant 还在为《崩坏:星…...

REIN方法:基于推理初始化的对话系统错误恢复技术

1. 项目概述REIN(Reasoning-Initialized Dialogue Error Recovery)是一种创新的对话系统错误恢复方法,它通过引入推理初始化机制来提升对话系统在出错时的自我修复能力。这个方法特别适合当前主流的基于神经网络的对话系统,能够有…...