成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

indexedDB事務(wù)功能的Promise化封裝

zombieda / 2061人閱讀

摘要:綜上,對(duì)進(jìn)行一定的封裝,來簡(jiǎn)化編碼操作?;膰L試對(duì)于這種帶大量回調(diào)的,使用進(jìn)行異步化封裝是個(gè)好主意。因此包括在內(nèi)的所有異步方法都會(huì)強(qiáng)制中止當(dāng)前事務(wù)。這就決定了一個(gè)事務(wù)內(nèi)部的所有操作必須是同步完成的。目前只實(shí)現(xiàn)了和,其他的有待下一步工作。

前言

本文是介紹我在編寫indexedDB封裝庫中誕生的一個(gè)副產(chǎn)品——如何讓indexedDB在支持鏈?zhǔn)秸{(diào)用的同時(shí),保持對(duì)事務(wù)的支持。
項(xiàng)目地址:https://github.com/woodensail/indexedDB

2015/11/12 注:這篇文章的思路有問題,大家看看了解一下就行,不要這么干。更好的做法已經(jīng)寫在了下一篇中。大家可以去看一下,順便幫忙點(diǎn)個(gè)推薦或者收藏一個(gè)。
地址:http://segmentfault.com/a/1190000003984871

indexedDB的基本用法
var tx = db.transaction("info", "readonly");
var store = tx.objectStore("info");
store.get("id").onsuccess = function (e) {
    console.log(e.target.result);
};

上面這段代碼中,開啟了一個(gè)事務(wù),并從名為info的store中取得了key為id的記錄,并打印在控制臺(tái)。
其中打印的部分寫在了onsuccess回調(diào)中,如果我們希望把取出的id加1然后返回就需要這樣:

// 方案1
var tx = db.transaction("info", "readwrite");
var store = tx.objectStore("info");
store.get("id").onsuccess = function (e) {
    store.put({key:"id",value:e.target.result.value + 1}).onsuccess = function (e) {
        ……
    };
};

// 方案2
var tx = db.transaction("info", "readwrite");
var store = tx.objectStore("info");
var step2 = function(e){
    store.put({key:"id",value:e.target.result.value + 1}).onsuccess = function (e) {
        ……
    };
}
store.get("id").onsuccess = step2;

前者用到了嵌套回調(diào),后者則需要將業(yè)務(wù)流程拆散。
綜上,對(duì)indexedDB進(jìn)行一定的封裝,來簡(jiǎn)化編碼操作。

Promise化的嘗試

對(duì)于這種帶大量回調(diào)的API,使用Promise進(jìn)行異步化封裝是個(gè)好主意。
我們可以做如下封裝:

function put(db, table, data ,tx) {
    return new Promise(function (resolve) {
        var store = tx.objectStore(table);
        store.put(data).onsuccess = function (e) {
            resolve(e);
        };
    });
}

var tx = db.transaction("info", "readwrite");
Promise.resolve().then(function(){
    put(db, "info", {……}, tx)
}).then(function(){
    put(db, "info", {……}, tx)
});

看上去這么做是沒有問題的,但是實(shí)質(zhì)上,在存儲(chǔ)第二條數(shù)據(jù)時(shí),會(huì)報(bào)錯(cuò)并提示事務(wù)已被停止。

事務(wù)與Promise的沖突

When control is returned to the event loop, the implementation MUST set the active flag to false.

——摘自W3C推薦標(biāo)準(zhǔn)(W3C Recommendation 08 January 2015)

如同上面的引用所說,目前的W3C標(biāo)準(zhǔn)要求在控制權(quán)回到事件循環(huán)時(shí),當(dāng)前開啟的事務(wù)必須被設(shè)置為關(guān)閉。因此包括Promise.then在內(nèi)的所有異步方法都會(huì)強(qiáng)制中止當(dāng)前事務(wù)。這就決定了一個(gè)事務(wù)內(nèi)部的所有操作必須是同步完成的。
也真是基于這個(gè)原因,我沒有在github上找到實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用的indexedDB封裝庫。
其中寸志前輩的BarnJS中到是有鏈?zhǔn)秸{(diào)用,然而只是實(shí)現(xiàn)了Event.get().then()。也就是只能一次數(shù)據(jù)庫操作,一次結(jié)果處理,然后就結(jié)束。并不能串聯(lián)多個(gè)數(shù)據(jù)庫操作在同一個(gè)事務(wù)內(nèi)。

不夠要實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用其實(shí)也不難,關(guān)鍵的問題就在于Promise本身是為異步操作而生的,因此會(huì)在鏈?zhǔn)秸{(diào)用的各個(gè)函數(shù)中返回事件循環(huán),從而減少網(wǎng)頁的卡頓。所以我們就需要實(shí)現(xiàn)一個(gè)在執(zhí)行每個(gè)函數(shù)過程之間不會(huì)返回事件循環(huán)的Promise,也就是一個(gè)同步化的Promise。

也許是這個(gè)要求太過奇葩,我沒發(fā)現(xiàn)網(wǎng)上有提供同步化執(zhí)行的promise庫。所以只能自己實(shí)現(xiàn)一個(gè)簡(jiǎn)單的。雖然功能不全,但也能湊活用了。下面是使用樣例和詳細(xì)代碼解釋,完整代碼見github。

使用樣例
// 這句是我封裝過后的用法,等效于:
// var tx = new Transaction(db, "info", "readwrite");
var tx = dbObj.transaction("info", "readwrite");

//正常寫法
tx.then(function () {
    tx.get("info", "a");
    tx.get("info", "b");
}).then(function (a, b) {
    tx.put("info", {key : "c", value : Math.max(a.v, b.v));
})

//偷懶寫法
tx.then(function () {
    tx.getKV("info", "a");
    tx.getKV("info", "b");
}).then(function (a, b) {
    tx.putKV("info", "c",  Math.max(a, b));
})
代碼解釋
var Transaction = function (db, table, type) {
    this.transaction = db.transaction(table, type);
    this.requests = [];
    this.nexts = [];
    this.errorFuns = [];
};
Transaction.prototype.then = function (fun) {
    var _this = this;
    // 若errored為真則視為已經(jīng)出錯(cuò),直接返回。此時(shí)后面的then語句都被放棄。
    if (this.errored) {
        return this;
    }
    // 如果當(dāng)前隊(duì)列為空則將自身入隊(duì)后,立刻執(zhí)行,否則只入隊(duì),不執(zhí)行。
    if (!_this.nexts.length) {
        _this.nexts.push(fun);
        fun(_this.results);
        _this.goNext();
    } else {
        _this.nexts.push(fun);
    }
    // 返回this以實(shí)現(xiàn)鏈?zhǔn)秸{(diào)用
    return _this;
};

Transaction的初始化語句和供使用者調(diào)用的then語句。

Transaction.prototype.put = function (table, data) {
    var store = this.transaction.objectStore(table);
    this.requests.push([store.put(data)]);
};
Transaction.prototype.get = function (table, key) {
    var store = this.transaction.objectStore(table);
    this.requests.push([store.get(key)]);
};
Transaction.prototype.putKV = function (table, k, v) {
    var store = this.transaction.objectStore(table);
    this.requests.push([store.put({k, v})]);
};
Transaction.prototype.getKV = function (table, key) {
    var store = this.transaction.objectStore(table);
    this.requests.push([store.get(key), item=>(item || {}).v]);
};

所有的函數(shù)都在發(fā)起數(shù)據(jù)庫操作后將返回的request對(duì)象暫存入this.requests中。
目前只實(shí)現(xiàn)了put和get,其他的有待下一步工作。另外,getKV和setKV是專門用于存取key-value數(shù)據(jù)的,要求被存取的store包含k,v兩個(gè)字段,其中k為主鍵。

// 該語句會(huì)在鏈?zhǔn)秸{(diào)用中的每個(gè)函數(shù)被執(zhí)行后立刻調(diào)用,用于處理結(jié)果,并調(diào)用隊(duì)列中的下一個(gè)函數(shù)。
Transaction.prototype.goNext = function () {
    var _this = this;
    // 統(tǒng)計(jì)前一個(gè)函數(shù)塊中執(zhí)行的數(shù)據(jù)庫操作數(shù)量
    var total = _this.requests.length;
    // 清空已完成數(shù)據(jù)庫操作計(jì)數(shù)器
    _this.counter = 0;
    // 定義全部操作執(zhí)行完畢或出差后的回調(diào)函數(shù)
    var success = function () {
        // 當(dāng)已經(jīng)有錯(cuò)誤出現(xiàn)時(shí),放棄下一步執(zhí)行
        if (_this.errored) {
            return;
        }
        // 將隊(duì)首的節(jié)點(diǎn)刪除,也就是剛剛執(zhí)行完畢的節(jié)點(diǎn)
        _this.nexts.shift();
        _this.requests = [];
        // 從返回的event集合中提取出所有result,如果有parser則使用parser。
        _this.results = _this.events.map(function (e, index) {
            if (_this.parser[index]) {
                return _this.parser[index](e.target.result);
            } else {
                return e.target.result;
            }
        });
        
        //判斷隊(duì)列是否已經(jīng)執(zhí)行完畢,否則繼續(xù)執(zhí)行下一個(gè)節(jié)點(diǎn)
        if (_this.nexts.length) {
            // 將節(jié)點(diǎn)的執(zhí)行結(jié)果作為參數(shù)傳給下一個(gè)節(jié)點(diǎn),使用了spread操作符。
            _this.nexts[0](..._this.results);
            _this.goNext();
        }
    };
    // 初始化events數(shù)組,清空parser存儲(chǔ)器
    _this.events = new Array(total);
    _this.parser = {};

    // 若該請(qǐng)求內(nèi)不包含數(shù)據(jù)庫操作,則視為已完成,直接調(diào)用success
    if (total === 0) {
        success();
    }

    // 對(duì)于每個(gè)請(qǐng)求將請(qǐng)求附帶的parser放入存儲(chǔ)區(qū)。然后綁定onsuccess和onerror。
    // 其中onsuccess會(huì)在每個(gè)請(qǐng)求成功后將計(jì)數(shù)器加一,當(dāng)計(jì)數(shù)器等于total時(shí)執(zhí)行回調(diào)
    _this.requests.forEach(function (request, index) {
        _this.parser[index] = request[1];
        request[0].onsuccess = _this.onsuccess(total, index, success);
        request[0].onerror = _this.onerror;
    })
};
Transaction.prototype.onsuccess = function (total, index, callback) {
    var _this = this;
    return function (e) {
        // 將返回的event存入event集合中的對(duì)應(yīng)位置
        _this.events[index] = e;
        _this.counter++;
        if (_this.counter === total) {
            callback();
        }
    }
};
Transaction.prototype.onerror = function (e) {
    // 設(shè)置錯(cuò)誤標(biāo)準(zhǔn)
    this.errored = true;
    // 保存報(bào)錯(cuò)的event
    this.errorEvent = e;
    // 一次調(diào)用所有已緩存的catch函數(shù)
    this.errorFuns.forEach(fun=>fun(e));
};
Transaction.prototype.cache = function (fun) {
    // 如果已設(shè)置錯(cuò)誤標(biāo)準(zhǔn)則用緩存的event為參數(shù)立刻調(diào)用fun,否則將其存入隊(duì)列中
    if (this.errored) {
        fun(this.errorEvent);
    } else {
        this.errorFuns.push(fun);
    }
};

核心的goNext語句以及success與error的回調(diào)。catch類似then用于捕捉異常。

總結(jié)

好累啊,就這樣吧,以后再加其他功能吧。另外這里面用了不少es6的寫法。所以請(qǐng)務(wù)必使用最新版的edge或chrome或firefox運(yùn)行?;蛘吣憧梢允謩?dòng)把es6的寫法都去掉。

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/86187.html

相關(guān)文章

  • indexedDB事務(wù)功能Promise封裝(二)——利用generator完成同步改造

    摘要:在不可以用的前提下,無論是同步化或者鏈?zhǔn)讲僮鞫加貌涣?。于是昨天我自己?shí)現(xiàn)了一個(gè)簡(jiǎn)單的同步執(zhí)行的,并以此為基礎(chǔ)實(shí)現(xiàn)了鏈?zhǔn)讲僮鳌? 前言 本來這個(gè)系列應(yīng)該不會(huì)這么快更新,然而昨晚寫完前一篇后才發(fā)現(xiàn)我的思路中有一個(gè)巨大的漏洞。導(dǎo)致我在前一篇中花費(fèi)大量時(shí)間實(shí)現(xiàn)了一個(gè)復(fù)雜的Transaction類——其實(shí)完全有更簡(jiǎn)單的方式來完成這一切。前篇:http://segmentfault.com/a/11...

    JackJiang 評(píng)論0 收藏0
  • html5 indexeddb簡(jiǎn)明api

    摘要:網(wǎng)上搜來一堆,,幾乎沒有找到滿意的答案,經(jīng)過匯總并結(jié)合自己的理解,封裝了一套簡(jiǎn)單的是一個(gè)異步對(duì)象,必須使用回調(diào)函數(shù)方式進(jìn)行調(diào)用打開一個(gè)數(shù)據(jù)庫,支持兩個(gè)參數(shù),第二個(gè)參數(shù)指定版本號(hào),我沒用到,讓瀏覽器自己創(chuàng)建版本號(hào)。 網(wǎng)上搜來一堆a(bǔ)pi,demo,幾乎沒有找到滿意的答案,經(jīng)過匯總并結(jié)合自己的理解,封裝了一套簡(jiǎn)單的api // indexedDB是一個(gè)異步對(duì)象,必須使用回調(diào)函數(shù)方式進(jìn)行調(diào)用 ...

    luckyw 評(píng)論0 收藏0
  • IndexedDB 簡(jiǎn)單封裝

    摘要:之前我在開發(fā)過程中使用的是,可以直接寫查詢數(shù)據(jù)。,用鍵值模式存儲(chǔ)數(shù)據(jù),而且就是專門為小數(shù)量數(shù)據(jù)設(shè)計(jì)的。只能是字符串而且空間有限。下面是自己看了阮一峰的文章簡(jiǎn)單的學(xué)習(xí)了下對(duì)這個(gè)瀏覽器數(shù)據(jù)庫有個(gè)大概的了解,下面是個(gè)人對(duì)簡(jiǎn)單的封裝。IndexedDB?瀏覽器數(shù)據(jù)庫,是一個(gè)非關(guān)系型數(shù)據(jù)庫,數(shù)據(jù)形式使用的是json,IndexedDB適合存儲(chǔ)大量數(shù)據(jù),它的API是異步調(diào)用的,當(dāng)然他的api?也相對(duì)復(fù)雜...

    Songlcy 評(píng)論0 收藏0
  • 【譯】漸進(jìn)式 Web App 離線存儲(chǔ)

    摘要:離線存儲(chǔ)數(shù)據(jù)的建議對(duì)尋址資源,使用這是的一部分。在到達(dá)儲(chǔ)量限制之前,兩種存儲(chǔ)機(jī)制都會(huì)一直進(jìn)行存儲(chǔ)。則沒有對(duì)存儲(chǔ)量做出限制,只是在之后會(huì)彈出提醒。是異步的基于回調(diào)函數(shù),但它同樣不支持。也是異步的基于回調(diào)函數(shù),在和中可以工作雖然使用的是同步。 拖拖拉拉好久,終于把個(gè)人博客整出來了。鳴謝 @pinggod。 厚著臉安利一下,地址是 http://www.wemlion.com/。歡迎訪問,歡...

    Joyven 評(píng)論0 收藏0
  • 【譯】漸進(jìn)式 Web App 離線存儲(chǔ)

    摘要:離線存儲(chǔ)數(shù)據(jù)的建議對(duì)尋址資源,使用這是的一部分。在到達(dá)儲(chǔ)量限制之前,兩種存儲(chǔ)機(jī)制都會(huì)一直進(jìn)行存儲(chǔ)。則沒有對(duì)存儲(chǔ)量做出限制,只是在之后會(huì)彈出提醒。是異步的基于回調(diào)函數(shù),但它同樣不支持。也是異步的基于回調(diào)函數(shù),在和中可以工作雖然使用的是同步。 拖拖拉拉好久,終于把個(gè)人博客整出來了。鳴謝 @pinggod。 厚著臉安利一下,地址是 http://www.wemlion.com/。歡迎訪問,歡...

    Charlie_Jade 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

閱讀需要支付1元查看
<