一句话结论:AI 对话历史别只存内存,刷新就没了;也别一股脑塞 localStorage,几十轮对话很快撑爆它那 5MB。IndexedDB 才是对的容器------异步、容量大(几百 MB)、能按会话分库查。我给一个移动端 AI 助手加离线缓存,用的就是它。
为什么非它不可?AI 对话有三个特点:消息多、单条可能很长(贴代码贴文档)、用户会回看历史。localStorage 同步读写会卡主线程,而且 JSON.stringify 整个历史每次写,数据一大就肉眼可见地顿。
封装一个极简的存取层
不引库,原生 IndexedDB 够用,就是 API 啰嗦。我封了几个 Promise 函数:
ini
function openDB() {
return new Promise((res, rej) => {
const req = indexedDB.open('ai-chat', 1);
req.onupgradeneeded = (e) => {
const db = e.target.result;
if (!db.objectStoreNames.contains('messages')) {
const store = db.createObjectStore('messages', { keyPath: 'id' });
store.createIndex('byConv', 'convId', { unique: false });
}
};
req.onsuccess = () => res(req.result);
req.onerror = () => rej(req.error);
});
}
async function saveMessage(msg) {
const db = await openDB();
return new Promise((res, rej) => {
const tx = db.transaction('messages', 'readwrite');
tx.objectStore('messages').put(msg);
tx.oncomplete = res;
tx.onerror = () => rej(tx.error);
});
}
keyPath: 'id' 让每条消息按自己的 id 做主键,byConv 索引让我能按会话 id 批量查。
按会话拉历史
回看某个会话时,走索引一次性拿出来:
ini
async function loadConv(convId) {
const db = await openDB();
return new Promise((res) => {
const tx = db.transaction('messages', 'readonly');
const idx = tx.objectStore('messages').index('byConv');
const out = [];
idx.openCursor(IDBKeyRange.only(convId)).onsuccess = (e) => {
const cur = e.target.result;
if (cur) { out.push(cur.value); cur.continue(); }
else res(out.sort((a, b) => a.ts - b.ts));
};
});
}
游标遍历完再按时间戳排序。IndexedDB 不保证返回顺序跟插入顺序一致,这个 sort 不能省------我省过一次,历史消息顺序乱了,被测试逮到。
流式消息怎么存?边流边存还是流完再存
这是真正的坑。AI 回复是一个 token 一个 token 流回来的。如果每个 chunk 都写一次 IndexedDB,几百次事务把性能拖垮;如果只在流结束写,中途用户切走应用、回来发现这条没了。
我的折中:内存里累积,每 500ms 落一次盘,流结束再补一次:
javascript
function streamSink(msgId, convId) {
let buf = '';
let timer = null;
return {
push(chunk) {
buf += chunk;
if (!timer) timer = setTimeout(flush, 500);
},
done() { clearTimeout(timer); timer = null; flush(); }
};
function flush() {
timer = null;
saveMessage({ id: msgId, convId, role: 'ai', content: buf, ts: Date.now() });
}
}
500ms 这个数我调过,太短了写太频繁,太长了崩溃丢得多。500 是体感平衡点。
一个没绕过去的小缺点
Safari 的隐私模式下 IndexedDB 可能直接抛错或静默失败,我加了降级:开库失败就退回纯内存模式,功能不挂只是刷新丢历史。这部分代码不优雅,但总比白屏强:
ini
let dbAvailable = true;
openDB().catch(() => { dbAvailable = false; });
后端对话能力我是在一个能拖拽搭智能体、不用写后端的平台上配的,前端只管把消息落进 IndexedDB。模型 API 直连讯飞 MaaS,没自己起服务。