概述IndexedDB 在并发或约束条件下可能产生事务失败。本文给出错误捕获与重试封装,并配合队列持久化提升稳定性。事务重试封装async function withRetry(fn, retries = 3) {
let attempt = 0;
while (true) {
try { return await fn(); } catch (e) {
attempt++;
if (attempt > retries) throw e;
await new Promise(r => setTimeout(r, 200 * Math.pow(2, attempt)));
}
}
}
function openDB(name) {
return new Promise((resolve, reject) => {
const r = indexedDB.open(name, 1);
r.onupgradeneeded = () => { const db = r.result; if (!db.objectStoreNames.contains('items')) db.createObjectStore('items', { keyPath: 'id' }); };
r.onsuccess = () => resolve(r.result);
r.onerror = () => reject(r.error);
});
}
async function putItems(records) {
return withRetry(async () => {
const db = await openDB('retrydb');
const tx = db.transaction('items', 'readwrite');
const store = tx.objectStore('items');
for (const r of records) store.put(r);
await new Promise((resolve, reject) => { tx.oncomplete = resolve; tx.onerror = () => reject(tx.error); tx.onabort = () => reject(tx.error); });
db.close();
});
}
失败队列与刷新function openQueue() {
return new Promise((resolve, reject) => {
const r = indexedDB.open('queue', 1);
r.onupgradeneeded = () => { const db = r.result; if (!db.objectStoreNames.contains('q')) db.createObjectStore('q', { autoIncrement: true }); };
r.onsuccess = () => resolve(r.result);
r.onerror = () => reject(r.error);
});
}
async function enqueue(payload) {
const db = await openQueue();
const tx = db.transaction('q', 'readwrite');
tx.objectStore('q').add(payload);
await new Promise((resolve, reject) => { tx.oncomplete = resolve; tx.onerror = () => reject(tx.error); });
db.close();
}
async function flushQueue(processor) {
const db = await openQueue();
const tx = db.transaction('q', 'readwrite');
const store = tx.objectStore('q');
const req = store.openCursor();
await new Promise((resolve, reject) => {
req.onsuccess = async e => {
const cursor = e.target.result;
if (cursor) {
try { await processor(cursor.value); store.delete(cursor.key); cursor.continue(); } catch { resolve(); }
} else { resolve(); }
};
req.onerror = () => reject(req.error);
});
await new Promise((resolve, reject) => { tx.oncomplete = resolve; tx.onerror = () => reject(tx.error); });
db.close();
}

发表评论 取消回复