摘要:綜上,對(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
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)化編碼操作。
對(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
摘要:在不可以用的前提下,無論是同步化或者鏈?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...
摘要:網(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)用 ...
摘要:之前我在開發(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ù)雜...
摘要:離線存儲(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/。歡迎訪問,歡...
摘要:離線存儲(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/。歡迎訪問,歡...
閱讀 2822·2021-11-16 11:44
閱讀 982·2021-10-09 09:58
閱讀 4508·2021-09-24 09:48
閱讀 4390·2021-09-23 11:56
閱讀 2417·2021-09-22 15:48
閱讀 1908·2021-09-07 10:07
閱讀 3214·2021-08-31 09:46
閱讀 519·2019-08-30 15:56