原文出处:https://developer.mozilla.org/en/IndexedDB/Using_IndexedDB(MOZILLA基金会开发者社区)
译者:黄俊文(本人其余博客http://user.qzone.qq.com/402487865)
译文如下:
IndexedDB是你将数据存储在用户浏览器上的一种新技术,它能让你创建的应用程序拥有更优异的数据查询性能,而且既能让你的应用程序在线上运行又能在离线状态下运行。
IndexedDB 包括一个同步和一个异步API。同步API计划交由WebWorkers 对象调用(但截至2011年12月份,IndexedDB还没有在Web Workers这个对象上得到实现,没有一款浏览器对此进行实现)。而异步的API计划让普通的网页使用(译者:使用window对象调用 IndexedDB API),在应用IndexedDB的绝大多数情况下我们都会用到它的异步调用的API,因此这篇文章将会讨论如何使用异步调用API。
关于这篇文章
这个教程主要围绕IndexedDB的动态调用API的使用,如果你对IndexedDB不是很熟悉,你可以先阅读 Basic Concepts About IndexedDB (IndexedDB的基础概念)这篇文章。
关于IndexedDB API的参考文档,你也可以阅读 IndexedDB 这篇文章以及它的子页面(译:译者的上一篇文章已经翻译,但子页面的没有进一步翻译),该文档 阐述了操作IndexedDB的对象,以及同步、异步API的接口说明和用法。
基本操作
IndexedDB的基本操作如下所示:
1.打开一个数据库和启动一个事务.
2.创建一个存储对象.
3.向数据库请求一些数据操作, 例如添加或者检索数据.
4.监听DOM事件来完成一些操作。
5.对结果集进行一些处理 (结果能通过请求对象获得).
OK,现在我们沿着这些基本操作为主线,逐步具体了解其中的细节信息。
创建和格式化存储对象
由于规范还在制定中,当前对IndexedDB的实现都是通过前缀的形式操作的。(浏览器生产厂商不希望不同浏览器对IndexedDB API的实现存在差异。)(译者:因此在w3c对HTML5规范还没有完全制定下来之前,浏览器厂商通常通过加前缀的方式给开发者试用,并反馈bug给浏览器厂商,当标准最终定稿和实现完善的时机浏览器厂商才会去掉前缀的方式支持该新特性)基于Gecko的浏览器使用moz前缀,而基于WebKit的浏览器则使用webkit前缀。在一些示例代码中,只需替换方法的前缀就可以达到对指定浏览器的支持(例如,如果你为Google的Chrome创建一个应用程序,使用webkitIndexedDB.open()替代mozIndexedDB.open())或者只需使用下一节所介绍的代码片段(译者:该片段能对不同浏览器返回对应的IndexedDB对象)。
连接数据库
一开始我们必须做以下操作:
- // 这样写能让代码更具可读性和更加简洁
- var indexedDB = window.indexedDB || window.webkitIndexedDB || window.mozIndexedDB || window.msIndexedDB;
- // 现在我们可以连接我们的数据库
- var request = indexedDB.open("MyTestDatabase");
看到了吗?连接一个数据库就跟其他操作一样-你要“请求” 它响应。
这个IndexedDB对象有一个很简单的方法,open(),当调用的时候,打开一个名为 "MyTestDatabase"的数据库。所有的IndexedDB的数据库都存储在同一个名字的数据源上,因此mozilla.com也许有一个名为“binky”的数据库,而另外mozilla.org也许也有一个完全不同的但名字却同为"binky"的数据库。(译者:这句没看懂。)如果该名字的数据库不存在,则创建它,如果已经存储了,就连接上这个数据库。
这个open请求如果遇到异常,那么它会返回一个IDBRequest 对象(这个对象和其他绝大多数IndexedDB异步方法一样)而不是一个open方法返回的值。
这个“open”方法允许传第二个参数,它允许更新数据库的设计。 那就是创建和删除存储的对象时候,如果数据库不是最新的,在这种情况下,后台就会自动触发onupgradeneeded 事件,该事件里的一个CHANGE_VERSION事务允许你出处理存储的对象。以下是一个打开一个指定版本号的数据库的写法。
- var request = indexedDB.open("MyTestDatabase", 3);
常规处理函数
处理绝大多数的包含请求操作程序中,你第一件事想起要做的莫过于添加成功请求后回调的处理函数,和不成功回调的处理函数。
- request.onerror = function(event) {
- // 对request.errorCode做一些需要的处理!
- };
- request.onsuccess = function(event) {
- // 对request.result进行一些需要的处理!
- };
onsuccess() 和 onerror()这两个函数是如何被调用的? 如果没有出现任何异常,那么标识成功调用的事件就会被触发(那么该事件的type属性会被设置为“success”,而且target属性会被设置为 request对象)(译者:DOM事件的type和target都属于2级DOM事件的标准属性,其中type属性指明该事件的名称,target则表示触发该事件的元素,即事件的目标节点)。一旦该事件被触发,request的onsuccess事件的处理函数就会被调用,并且该函数接收一个type 属性为success的事件对象作为参数。另外,如果请求过程中遇到任何异常的话,request对象中type属性为“error”的事件就会被触发,该事件的处理函数会被传入一个type属性为error的事件对象作为参数。
IndexedDB API最小化出错处理程序相关的API,因为你不希望看到太多的出错事件(至少,当你才刚开始使用IndexedDB API的时候是不太愿意看到的)。例如在你打开一个数据库这样的基本操作,你是不愿意看到太多的出错事件;但是API提供了一些常见异常发生所需要的条件。最常见的问题之一就是用户选择不授权你开发的网络应用程序去创建一个数据库。其中IndexedDB最大的用处在于允许你存储大量的数据以供在离线的状态下使用。(去获得更多不同浏览器对本地存储的最大配额,请查阅存储限额)(译者:Google的chrome浏览器分配最大5M的存储空间)。
当然,浏览器是不想允许一些网络广告和恶意站点去损害你的计算机,所以浏览器在任何网络应用程序在尝试打开一个IndexedDB存储的第一时间都会事先提示用户,用户可以选择允许也可以选择拒绝授权应用程序使用本地的IndexedDB 存储,同样,当浏览器被设置为隐私模式,则IndexedDB也是不能使用的,(FireFox浏览器通过工具-》进入隐私浏览模式设置,Chrome浏览器设置匿名模式)。隐私浏览方式的主旨在于不产生任何浏览痕迹,所以在这种隐私模式下是不能打开IndexedDB数据库的。
现在,假设浏览器用户允许你去创建一个数据库,而你又获得一个成功请求打开数据库的事件并回调成功请求后相应的处理程序;接下来会怎样呢?indexedDB.open()执行完后会返回一个request对象,所以request.result是一个IDBDatabase的一个实例,并且紧接着你无疑会想办法去存储它。那么你的代码可能会类似如下所示:
- var db;
- var request = indexedDB.open("MyTestDatabase");
- request.onerror = function(event) {
- alert("Why didn't you allow my web app to use IndexedDB?!");
- };
- request.onsuccess = function(event) {
- db = request.result;
- };
异常处理
如以上提到的,这些异常事件的目标节点都是指向出现请求异常的请求对象,然后这些异常事件经过事件冒泡,最终会传到数据库对象上,如果你想避免为每次进行打开数据库的请求(如上面的request对象)都定义一次异常处理程序,那么你可以选择在数据库对象上监听异常事件,如下所示:
- db.onerror = function(event) {
- // 针对这个数据库的所有请求的异常都统一在这个异常事件处理程序中完成。(译者:这样做有效减少监听的数量,在事件冒泡过程的最顶层实现一次监听就可以了,这也是提高JS运行效率的一种技巧。)
- alert("Database error: " + event.target.errorCode);
- };
当打开一个数据库时最常遇到的其中一种错误是VER_ERR。这个异常,说明存储在磁盘中的数据库的版本比你现在尝试访问的数据库要更高,所以这种出错情况必须在异常处理程序中进行处理。
更新数据库的版本
现在你已经创建了一个数据库,你需要对数据库进行设计结构。很多SQL发烧友可能会认为IndexedDB数据库包含了很多不同可供数据库使用的数据表。而 IndexedDB对此进行了简化(译者,有些书籍称IndexedDB或Web SQL之类的数据库为简易数据库,但Web SQL已经出现技术性滑铁卢,W3C已经将重心转移到IndexedDB上,并且FirFox和Webkit系列的浏览器相继实验性地支持 IndexedDB的异步调用API),但你依然需要去建立你的数据库。
当你打开数据库的时候,你标识你所期望的数据库版本号,如果你尝试打开的数据库的版本号跟你期望的不一致的话,那onupgradeneeded 这个数据库对象的更新版本事件的处理程序就会被调用。这个处理程序非常重要,因为这个处理函数是你的代码当中唯一可以让你更新存储的对象。
示例代码如下:
- // 截止2012-02-22为止,WebKit内核的浏览器还没有对onupgradeneeded进行实现.
- request.onupgradeneeded = function(event) {
- // 更新存储的对象(译者,此处不懂翻译,原文为Update object stores and indices)
- ....
- }
WebKit到目前为止还没有实现获取指定版本号的本地数据库的功能,同时更谈不上出现版本号不一致时触发的处理事件的支持,想获取更多信息到Chromium bug 108223(http://code.google.com/p/chromium/issues/detail?id=108223).如想获取更多Webkit更新数据库版本号的资料请阅读IDBDatabase 相关文章。
数据库设计
现在我们去设计数据库。IndexedDB 使用对象存储而不是数据表,并且一个简单的数据库能包含任意多的存储对象。无论在什么时候,当对一个值进行存储的时候,都必须关联到一个键值。有几种方式获取键,这些方式分别取决于存储对象使用键路径(https://developer.mozilla.org/en/IndexedDB#gloss_key_path)还是键×××(https://developer.mozilla.org/en/IndexedDB#gloss_key_generator)。
以下说明了设置键的几种方式。
你还可以在任何存储对象上限定存储的总个数,控制存储对象能存储多少个对象,索引能让你通过存储对象的属性值去搜索记录而不是通过对象的关键字(译者:关键字是必须唯一,而索引可以唯一也可以不唯一,将经常需要查询的字段设置为索引会提高查询效率,另外可以达到确保数据的正确性。)。
加上, 索引针对存储的数据做一些简单的约束限制,如设置了唯一索引的属性,则录入的数据中不能出现任意两条数据的该属性值出现相同的情况,例如,如你有一个存储对象专门存储人的信息,而你想确保每个人都有唯一的email地址,则你应该使用唯一索引来约束email这个属性。
以上听起来不好理解,但以下这个简单的例子应该能更好说明概念:
- // 这就是我们客户数据的结构.
- const customerData = [
- { ssn: "444-44-4444", name: "Bill", age: 35, email: "bill@company.com" },
- { ssn: "555-55-5555", name: "Donna", age: 32, email: "donna@home.org" }
- ];
- const dbName = "the_name";
- var request = indexedDB.open(dbName, 2);
- request.onerror = function(event) {
- // 异常处理.
- };
- request.onupgradeneeded = function(event) {
- var db = event.target.result;
- // 创建一个名为“objectStore”的对象来存储我们的客户数据。
- // 我们选择使用“ssn”作为我们的key path,因为它能保证唯一
- var objectStore = db.createObjectStore("customers", { keyPath: "ssn" });
- // 创建一个通过name属性去查找客户数据的索引。因为可能出现
- // 出现重复,所以我们不能使用唯一索引.
- objectStore.createIndex("name", "name", { unique: false });
- // 我们想确保任意两个客户的email是不同的,
- // 所以我们用了唯一索引
- objectStore.createIndex("email", "email", { unique: true });
- // 用我们刚新建的objectStore来存储客户数据.
- for (i in customerData) {
- objectStore.add(customerData[i]);
- }
- };
如之前所提到的,onupgradeneeded 处理函数是唯一个地方允许你修改数据库的结构和设计。在里面,你可以新建和删除存储的对象并可设置和移除指数。(译:本人还没理解原文所指的indices具体是指什么)
存储对象(译者:上例中存储客户数据表的objectStore对象)是通过简单地调用createObjectStore()函数创建的,这个方法有一个标识存储对象名称的参数和一个对象参数(译者:如JSON表示的对象)。尽管对象参数是可选的,但它起到相当重要的作用,因为它让你定义一些重要的可选属性同时你也可以约束你想要创建的存储对象的类型。在我们上面的例子中,我们请求创建一个名字为“customer”的存储对象和定义了关键路径,这个关键路径是唯一标识每一条存储记录的。在上面这个例子中,这个关键路径就是“ssn”,全称为社会安全号码(译者,可以理解成×××号),这个号码要保证唯一的,每个存储记录的 “ssn”属性必须能够唯一标识这条数据。
我们已经请求了一个名为“name”的索引,这个索引对应存储记录的name属性。与createObjectStore()这个函数一样,createIndex()都有一个可选的对象参数,这个对象参数用于约束你所要创建的索引。
现在,我们可以通过“ssn”属性直接取回我们存储的对应的客户数据记录,或者通过使用name索引来获取数据,如想了解是如何获取数据的,你可以阅读使用索引部分。
添加和移除数据
在你对新建的数据库进行任何操作之前,你需要开启一个事务,该事务来自数据对象,并且你必须为你想要操作的存储对象(译者:IDBDatabase实例)指定一个事务类型涉及到哪些存储对象(译者:通过createObjectStore方法参数中设置的存储对象的名称来表示)。同时,你也要决定你是想要修改数据库还是只是需要读取数据。虽然事务有3种类型(read-only只读, read/write读写, 和 snapshot 快照),但你最好还是使用read-only只读类型如果可以的话,因为这些事务能同时进行。
添加数据到数据库
如果你已经创建了一个数据库,那么你可能想向它写入数据,代码如下:
- var transaction = db.transaction(["customers"], IDBTransaction.READ_WRITE);
transaction() 函数有三个参数,其中有两个是可选的。第一个参数是存储对象的一个数组(译者:该数组的元素跟db.createObj ectStore方法的第一个参数概念相当),用于指明这个事务作用于哪些存储数据,如果你想该事务能作用于所有存储对象(译者,这里的存储对象大概相当于我们SQL二维数据库的数据表的角色),则你可以用空数组作为这一参数的值,如果你不想给第二个参数设置任何东西,那默认会是一个只读事务。如果你想事务能写入,那么你需要使用READ_WRITE标识符。
当你创建了一个事务你需要了解它的生命周期。事务性能与事件循环密切相关。如果当你没有操作这个事务并且将它返回给事件冒泡的环路上,则事务会处于睡眠状态,要激活它的唯一途径就是向事务发出请求。当你完成请求后你会得到一个DOM事件,假设这个请求成功,你还可以在回调函数里做一些对该事务的扩展,当你没有对事务进行扩展则就返回给事件冒泡环路,则事务依然将处于睡眠状态,等等。实际上只有一直请求来维持事务的激活状态。事务的生命周期其实好简单,但它可能需要你一些时间来熟悉它和习惯它。看多一些例子同样有所帮助。当你把一些东西弄错的时候,你可以开始检查RANSACTION_INACTIVE_ERR标识符所显示的错误代码片段以供找出错误的根源。
事务可以监听3种不同类型的DOM事件:error(异常),abort(中止)和complete(完成)。我们已经讨论了异常事件的冒泡了,如果请求事务遇到异常,那么这个事务会接收到异常的事件通知。其中比较隐蔽的点在于当请求事务的异常出现的时候,这个异常有个默认行为,就是去中止事务。除非你对异常事件对象执行preventDefault()方法来阻止其默认行为的发生,那么整个事务将会回滚(译者:也就是回到最初的修改前的状态)。这种设计的目的在于让你去思考当出现异常的时候该进行如何的处理方式,但你可以添加一个可以处理各种请求异常的归总处理程序(利用事件冒泡,在冒泡阶段的最顶层设置监听),如果对冒泡阶段的每一层都实现监听异常,那将会显得很笨重。如果你不想对异常事件做任何处理,你可以对事务调用abort()来中止事件,那时事务将会回滚并且中止事件会被事务对象所触发。另外,当所有的请求都完成的时候,你将会接受到一个complete完成事件通知。如果你做了很多数据库操作,那么去跟踪事务而非个别的请求,当然这个也是取决于你个人的意志了。
现在创建了一个事务,你将会希望从该事务获取存储对象。当创建一个你指定了存储对象的事务只能让你拥有一个存储对象而已,然后你可以往里面添加你需要的数据。
- // 当数据库被填充数据后要做的事情.
- transaction.oncomplete = function(event) {
- alert("All done!");
- };
- transaction.onerror = function(event) {
- // 不要忘了处理异常!
- };
- var objectStore = transaction.objectStore("customers");
- for (var i in customerData) {
- var request = objectStore.add(customerData[i]);
- request.onsuccess = function(event) {
- // event.target.result == customerData[i].ssn
- };
- }
上述的例子中event.target.result的result是customerData[i]这条数据对应的一个键。所以在这种情况下,result相当于customerData[i] 的“ssn”属性,因为在数据库的设计时定义了该属性为数据表的关键路径(译者:相当于关键字段)。注意add()方法要求添加的对象的关键字段的值不会与数据库中已存在的记录存在相同的值。如果你尝试修改已经存在的实体,或者你不关心添加的数据的关键字段是否已经存在于数据库表中,那么你需要使用 put()方法。(译者:但是这里是否会覆盖原来的记录还是额外添加一条重复数据,则有待实验。)
从数据中移除数据
移除数据是非常简单的:
- var request = db.transaction(["customers"], IDBTransaction.READ_WRITE)
- .objectStore("customers")
- .delete("444-44-4444");
- request.onsuccess = function(event) {
- // 删除操作完成!
- };
从数据库中获取数据
现在数据库中含有一些信息,你可以有几种方式进行获取。第一种,简单调用get()函数。你需要提供键来获取对应的值,类似于以下所示:
- var transaction = db.transaction(["customers"]);
- var objectStore = transaction.objectStore("customers");
- var request = objectStore.get("444-44-4444");
- request.onerror = function(event) {
- // 异常处理!
- };
- request.onsuccess = function(event) {
- // 对request.result进行一些处理(译者:request.result相当于记录对象)!
- alert("Name for SSN 444-44-4444 is " + request.result.name);
- };
如上所示其实简单的调用还是需要挺多的代码,这里我们展示如何对它进行进一步的简化,而且建议你在数据库的水平上对异常进行处理.。
- db.transaction("customers").objectStore("customers").get("444-44-4444").onsuccess = function(event) {
- alert("Name for SSN 444-44-4444 is " + event.target.result.name);
- };
明白它是如何工作的吗?只有唯一一个存储对象,你可以不需要传一个数据列表给它,而只需传一个该数据列表的一个名字并以字符串的形式传给它就行了。如果你只需从数据库中读取,那么你不需要创建事务的时候引入READ_WRITE 标识参数,如果你没有指定操作模式的方式调用transaction()那么你创建的事务将会是只读的模式的。另一个比较微妙的地方就是你无需对你请求的记录保存为一个可变类型的对象,而是通过DOM事件会在后台被传入一个请求对象,而那个事件对象的target属性的result属性就是你要获取的记录对象,是不是很简单?!
使用游标
使用get()方法需要你提供你想要获取记录的关键字。如果你想对存储对象存储的所有记录一条一条读出来的话,类似遍历,那么你可以使用游标。代码如下所示:
- var objectStore = db.transaction("customers").objectStore("customers");
- objectStore.openCursor().onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- alert("SSN关键字 " + cursor.key + " 对应的值 " + cursor.value.name);
- cursor.continue();
- }
- else {
- alert("没有更多的实例了!");
- }
- };
这个openCursor()方法需要几个参数。第一个,你可以通过键变化幅度对象来指定记录的最大。第二个,你可以指定游标遍历的顺序,默认为升序(译者,如上面的示例代码所示)。在上述例子中,我们升序方式循环读取所有记录对象。这个success回调函数有一些特殊。游标本身就是请求对象的 result对象(以上我们使用了简写,所以为event.target.result)。那么我们实际需要的键,值则可以通过游标对象的key和 value属性获取。如果你想继续通过游标获取一条记录的信息,你还得对游标对象调用continue()方法。当你已经获取玩最后一条记录(或者已经没有实体数据符合你openCursor()方法请求的要求)程序依然会执行success回调函数,但这个result属性的值为undefined。
一种使用游标的常见模式就是获取存储对象中所有记录和将这些记录添加到一个数组,如下所示:
- var customers = [];
- objectStore.openCursor().onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- customers.push(cursor.value);
- cursor.continue();
- }
- else {
- alert("已经获取所有客户的资料: " + customers);
- }
- };
警告: 以下函数并不是IndexedDB标准的一部分!
Mozilla基金会自己独自实现了getAll() 去处理上一例子的情况(译者:其实游标的实现效率是明摆着是低下的,一般在数据库的设计当中应尽量避免使用游标,能不用的地方不要使用,估计是这样的原因,Mozilla基金会专门提供了一个getAll()方法替代了相当一部分游标才能做的事,给开发者更多选择的余地,也是提高其自家浏览器运行性能的一个举措吧)。但它并不是IndexedDB的标准,所以它可能会在未来的版本中废弃。我们在这里把这个方法引入进来,是因爲我们认为它是有一定用处的。以下代码很明显与上述例子做了同样的事情:
- objectStore.getAll().onsuccess = function(event) {
- alert("已经获取所有客户的资料: " + customers);
- };
游标的应用要关联到游标的value属性上,因为对象都是惰性创建的(译者,就是你使用到该属性,后台才去把记录的对象赋值给这个value属性并非一次性获取全部记录),Gecko内核的浏览器一定是一次性创建对象的。如果你只是希望查看所有的键,那么使用游标肯定是比使用getAll()要高效(译者:原因好简单,因为getAll()将值都一次性获取的,而这不是我所希望查看的键,这显得是多余的,而游标则是通过value后期需要用到的时候才去获取这些记录的而并没有一次性获取我不希望查看的东西),但如果你将所有记录以数组形式获取出来则使用getAll()方法显得更高效(译者:游标后期获取记录的方式造成相当的性能损耗,没有一次性获取来得高效)。
使用索引
存储客户数据的时候使用”SSN“作为关键字段是符合逻辑的,因为”SSN“唯一标识一个个体。(先不管这个是否涉及到个人隐私的问题,因为这个已经超出了本文的讨论的范围。)如果你需要通过客户的名称来查找客户资料,你需要对每个记录进行搜索直到你找到正确的那一个。如果按照这样的搜索方式搜索则会变得效率低下,那么你可以使用索引来代替这种效率低下的方式。
- var index = objectStore.index("name");
- index.get("Donna").onsuccess = function(event) {
- alert("名字叫Donna's SSN字段的值为 " + event.target.result.ssn);
- };
这个“name”游标并不是唯一的,所以很有可能会有多于一条的记录的name字段值都是“Donna”的情况。在这种情况下,你通常会获得键值最低的那一条记录。如果你需要所有的记录都有一个name属性则你可以使用游标。你可以打开两个不同类型的游标。普通的游标指向的记录包含所有记录该有的属性。而另外一种游标指向的记录就只有关键字段的值。这个区别如下面的代码所示:
- index.openCursor().onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- // cursor.key 是键的名字, 类似 "Bill", 和 cursor.value 就是一个完整的记录对象.
- alert("Name: " + cursor.key + ", SSN: " + cursor.value.ssn + ", email: " + cursor.value.email);
- cursor.continue();
- }
- };
- index.openKeyCursor().onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- // cursor.key 是键的名字, 类似 "Bill", and cursor.value 就是 “SSN” 关键字段的值.
- // 没办法直接从cursor游标对象中获取记录余下的属性.
- alert("Name: " + cursor.key + ", "SSN: " + cursor.value);
- cursor.continue();
- }
- };
设置游标(译者:游标与索引结合使用)读取数据的顺序(升序、降序)和指定范围
如果你想你的游标能对记录进行一些筛选,你可以使用关键字范围对象并且将这个对象作为openCursor()方法或者openKeyCursor()方法的第一个参数。你可以对这个关键字范围对象设置只获取匹配某个条件的值,或者在某个值之前或之后的所有记录,并且可以控制是否包含这些边界值上的记录(译者:如果设置true则不包括处于边界值上的记录,false则包括边界值上的记录),“close”(译者,对应下面的lowe rBoundKeyRange等 ),(说明,这个关键字范围对象包含给定的边界值)而“open”(译者,对应下面的lowerBoundOpenKeyRange等 )(说明,这个关键字范围对象将不包括给定的边界值)。它的工作原理如下:
- // 只匹配 "Donna"
- var singleKeyRange = IDBKeyRange.only("Donna");
- // 匹配"Bill"之前的所有记录, 并且包括"Bill"
- var lowerBoundKeyRange = IDBKeyRange.lowerBound("Bill");
- // 匹配"Bill"之前的所有记录, 但不包括"Bill"
- var lowerBoundOpenKeyRange = IDBKeyRange.lowerBound("Bill", true);
- // 匹配“Donna”之后的所有记录, 但不包括 "Donna"
- var upperBoundOpenKeyRange = IDBKeyRange.upperBound("Donna", true);
- // 匹配在"Bill" 和 "Donna"之间的所有记录,包括“Bill”,但不包括“Donna”
- var boundKeyRange = IDBKeyRange.bound("Bill", "Donna", false, true);
- index.openCursor(boundKeyRange).onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- // 对满足游标匹配标准的记录进行相关的处理。
- cursor.continue();
- }
- };
有时你更想降序遍历对象而不是升序(升序为默认的游标排序方式)。降序的实现是通过传递一个PREV标识符去openCursor()方法中:
- objectStore.openCursor(null, IDBCursor.PREV).onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- // 对记录实例做一些处理.
- cursor.continue();
- }
- };
因为“name”索引不是唯一的,所以有可能存在多条记录拥有相同的“name”属性值。但要注意关键字却不允许出现这种类似的情况,它必须是唯一标识一条记录的。如果你希望游标去过滤掉具有重复的索引值的记录,那么你可以选择传递一个NEXT_NO_DUPLICATE标识作为这个游标的顺序参数,当然如果是降序并且又要去掉重复的索引值的记录要使用PREV_NO_DUPLICATE标识,关键字最小的那个总是第一个返回(译者:如果是降序,则肯定是总是最大的关键字优先被返回的)
- index.openKeyCursor(null, IDBCursor.NEXT_NO_DUPLICATE).onsuccess = function(event) {
- var cursor = event.target.result;
- if (cursor) {
- // 对记录实例做一些处理.
- cursor.continue();
- }
- };
当一个web应用程序被打开于另一个选项卡时版本的更新
当你的应用程序对数据库请求的版本发生变化的时候,你要周到地考虑到你的用户有可能在另一个选项卡中正在打开一个使用旧版本数据库的页面。当你调用 open()方法去创建一个获取版本更高的数据库的时候,那么在你开始对数据库进行任何变动之前,一定要对其他还在使用旧版本数据库的页面发出明确的通知,即通知那些使用旧版本数据库的页面告知用户需要关闭该页面等相关处理。(译者:但如何知道其他页面都已经关闭呢,好让我当前请求更高版本数据库的页面进行初始化新版本数据库的操作呢?本人还在思考这个问题。)
- var openReq = mozIndexedDB.open("MyTestDatabase", 2);
- openReq.onblocked = function(event) {
- // 当其他的选项卡都加载了数据库时,那么数据库要先关闭。
- // 在我们开始着手之前.
- alert("请将打开此站点的其他浏览器选项卡都关闭掉!");
- };
- openReq.onupgradeneeded = function(event) {
- // 所有其他的数据库已经关闭. 初始化最基本的东西.
- db.createObjectStore(/* ... */);
- useDatabase(db);
- }
- openReq.onsuccess = function(event) {
- var db = event.target.result;
- useDatabase(db);
- return;
- }
- function useDatabase(db) {
- // 当另一个页面请求一个版本发生变化的时候,要确保已经添加一个处理程序去监听这种请求
- // .我们必须首先关闭数据库(译者:通过不关闭页面来达到的)。这种操作允许页面去获取更新版本的数据库。
- // 只有用户关闭这些选项卡才能让数据库的更新生效。
- db.onversionchange = function(event) {
- db.close();
- alert("这个页面的新版本已经就绪。请重新加载(刷新)!");
- };
- // 对数据做一些处理.
- }
安全性
IndexedDB 使用了同源策略,就是说IndexedDB 的数据库捆绑与创建它的原始站点是捆绑在一起(相当于,这就是站点的一个域或一个子域),所以它不能被其他站点所使用。还有一个很重要的点就是注意到在框架页(不管是<frame>和<iframe>都不能够)中引入非原始站点都不能使用到原始站点创建的IndexedDB数据库。这是一个处于对安全性的考虑(译者:从机制层面上杜绝数据的随意窃取。)看bug 595307(https://bugzilla.mozilla.org/show_bug.cgi?id=595307)。
进阶
如果你想去开始调整API,请跳转到相关文档 和查阅不同的方法。
参考文献
相关文献
IndexedDB 相关API
IndexedDB相关的API规范(http://www.w3.org/TR/IndexedDB/)1
结语:
本篇外文翻译历时一周,内容相当多,比较全面介绍了IndexedBD的基础概念和应该注意的地方,由于本人的能力水平有限,翻译出错也是在所难免,欢迎指正。
翻译本文初衷,一开始想了解localStorage来存储图片的技术,后来发现结合IndexedDB会是一个不错的选择,后来就想先研究一下这个 IndexedDB的概念,发现其已经得到w3c的重点研发对象,将用于取代之前的Web SQL,因此了解它变得非常有价值,尤其作为一名前端开发者,这篇文章也是本人当月的重点技术博文,希望对前端开发的同行也有所帮助,谢谢阅读!