javascript - 在不更改版本的情况下从 IndexedDB 阻塞事件中恢复

标签 javascript indexeddb

我正在尝试针对 IndexedDB 库编写一些测试。

这样做时,我故意通过使用更高版本针对已打开的数据库调用 open() 来创建一个 blocked 事件。

虽然我可以通过在第一个 open() 期间在早期版本上设置的 versionchange 监听器中关闭数据库来规避阻塞,但出于测试目的,我会希望避免添加此 versionchange 监听器,而是让 blocked 事件完全触发,以便我可以确认我的 onblocked 监听器是否正常工作。

问题在于,在此阶段调用关闭数据库似乎无助于解除阻塞,随后删除同一数据库的调用也会收到 blocked 事件(在 Firefox 和 Chrome )。因此,在执行此阻止代码后,我无法继续进一步测试。

如果不使用之前添加的 versionchange 监听器,是否无法从 blocked 事件中恢复?

代码如下:

var DB_NAME = 'test', DB_VERSION = 1, NEW_DB_VERSION = 2;
indexedDB.deleteDatabase(DB_NAME);

var data = [{
    firstName: 'Brett',
    lastName: 'Zamir',
    age: 43
}, {
    firstName: 'Jane',
    lastName: 'Doe',
    age: 40
}, {
    firstName: 'Bob',
    lastName: 'Doe',
    age: 42
}];

var req = indexedDB.open(DB_NAME, DB_VERSION);
req.onerror = function (evt) {
    console.error("openDb:", evt.target.errorCode);
};
req.onsuccess = function (evt) {
    var db = this.result;

};
req.onupgradeneeded = function (e) {
    var db = e.target.result;
    var os = db.createObjectStore('test', {
        keyPath: 'id',
        autoIncrement: true
    });
    os.transaction.oncomplete = function () {
        var testTx = db.transaction('test', 'readwrite');
        testTx.oncomplete = function () {
            console.log('storage completed');
            fireBlockedEvent(db);
        };
        var testStore = testTx.objectStore('test');
        for (var i in data) {
            testStore.add(data[i]);
        }
    };
};

function fireBlockedEvent (db) {
    var req = indexedDB.open(DB_NAME, NEW_DB_VERSION);
    req.onblocked = function () {
        console.log('blocked');
        db.close();
        var req = indexedDB.deleteDatabase(DB_NAME);
        req.onsuccess = function () {
            console.log('we ok');
        };
        req.onblocked = function () {
            console.log('still blocked');
        };
    };
}

最佳答案

您正在打开两个连接,但只关闭一个。

fireBlockedEvent 中,您将关闭通过 open(DB_NAME, DB_VERSION) 打开的连接(作为 db 传入)。但在调用 deleteDatabase() 之前,您永远不会通过 open(DB_NAME, NEW_DB_VERSION) 关闭在 fireBlockedEvent 本身中打开的连接。

删除请求将被阻止,直到所有连接都关闭。

由于您正在编写测试代码,因此很难在不改变语义的情况下提出具体的修复建议,但这可行:

function fireBlockedEvent (db) {
    var req = indexedDB.open(DB_NAME, NEW_DB_VERSION);
    req.onblocked = function () {
        console.log('blocked');
        // close the blocking connection:             
        db.close();
    };
    req.onsuccess = function (e) {
        // close the formerly blocked connection:
        e.target.result.close();

        var req = indexedDB.deleteDatabase(DB_NAME);
        req.onsuccess = function () {
            console.log('we ok');
        };
        req.onblocked = function () {
            console.log('still blocked');
        };
    };
}

关于javascript - 在不更改版本的情况下从 IndexedDB 阻塞事件中恢复,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33441956/

相关文章:

javascript - 无法将日期选择器应用于使用 .load 添加到应用程序的 html 页面上的字段

JavaScript:获取和设置 URL 哈希参数?

javascript - 如何过滤数组内的对象?

jquery - HTML 表格中堆叠和分组的条形图

javascript - 将配置文件文件夹移动到 sdcard firefox mobile 14.0

javascript - 在导航栏组件中组合 bootstrap-vue 和 vue-router

windows-store-apps - Windows 应用商店应用提交的 IndexedDB 问题

javascript - 将 IndexedDB 中的数据拉入数组并通过 ReactJS 输出

javascript - IndexedDB:如何将多个索引与 'multiEntry: true' 一起使用

javascript - 提高大规模 IndexedDB 插入的性能