在现代浏览器的本地存储方案中,indexedDB 是一项重要的能力组成,它是可以在浏览器端使用的本地数据库,可以存储大量数据,提供接口来查询,还可以建立索引,这些都是其他存储方案 Cookie 或者 LocalStorage 无法提供的能力 。单从数据库类型来看,IndexedDB 是一个非关系型数据库(不支持通过 SQL 语句操作) 。
IndexedDB 的主要概念
IndexedDB 是一个比较复杂的 API 组合,学习它的过程就相当于学习它的各个对象 API 接口,包括以下这些( IDB 指当前操作的数据库实例 ):
- 数据库:IDBDatabase 对象
- 仓库对象: IDBObjectStore 对象
- 索引:IDBIndex 对象
- 事务:IDBTransaction 对象
- 操作请求:IDBRequest 对象
- 指针:IDBCursor 对象
- 主键:IDBKeyRange 对象
在这些 API 中包含一些主要概念:
- 数据库:数据库是所有相关数据的基本容器 。在同源策略( 协议 + 域名 + 端口 )的前提下,每个域名下可以新建任意多的数据库 。IndexedDB 中有版本概念,这就规定了同一时刻下只有一个版本的数据库存在 。
- 对象仓库:对象仓库 ObjectStore 在 IndexedDB 中对应的是 MySQL 中的表 Table 。
- 数据:对象仓库中记录的是若干条数据,数据只有主键和数据体两个部分,主键不能重复,可以为自增的整数编号或者数据中指定的一个属性 。数据体可以是任意数据类型,不限于对象 。
- 索引:为不同的属性建立索引可以加快数据的检索 。
- 事务:数据的 CURD (增删查改) 都要通过事务来完成 。
通过简单的对比图来理解 IndexedDB 的概念:

文章插图
快速起步 IndexedDB
在介绍了 IndexedDB 的主要概念之后,可以通过一个简单实用的 CURD 例子来学习在日常开发中我们是怎么使用 IndexedDB 的,各个 API 细节日后可以慢慢深入学习 。
- 必不可少的浏览器支持检查:
if(!('indexedDB' in window)){ console.log('当前浏览器支持 IndexedDB'); return;} else { console.log('您的浏览器不支持 IndexedDB') // todo 建议升级或者更换其他浏览器}
- 连接数据库
// 数据库实例let db;// 数据库打开操作,第一个参数是数据库名称, 第二个参数是数据库版本let DBRequestLink = window.indexedDB.open('dataBaseName', 1)DBRequestLink.onsuccess = function(event) { // 获取数据库实例 db = DBRequestLink.result; // 其他操作};// 这个监听回调触发于数据库首次新建、open数据库时传递新版本(只能比之前传递的版本高)DBRequestLink.onupgradeneeded = function(event) {};
- 创建数据库的主键和字段
DBOpenRequest.onupgradeneeded = function(event) { let db = event.target.result; // 创建一个数据库存储对象,并指定主键 let objectStore = db.createObjectStore('person', {keyPath: 'id', autoIncrement: true }); /* 定义存储对象的数据项* 第一个参数是创建的索引名称,可以为空 * 第二个参数是索引使用的关键名称,可以为空 * 第三个参数是可选配置参数,可以不传,常用参数之一就是 unique,表示该字段是否唯一,不能重复 */objectStore.createIndex('id', 'id', { unique: true}); objectStore.createIndex('name', 'name'); objectStore.createIndex('age', 'age'); objectStore.createIndex('sex', 'sex');};
- 在上述操作中,我们先定义了上文中提到的 IDBObjectStore 对象,并指定主键为 id,随后又通过 createIndex 来创建字段 。值得注意的是虽然创建了四个字段,但在 IndexedDb 中数据还是分为主键 id 和数据主体两个部分,并不会像 MYSQL 中在 Table 中呈现四列 。
- 向数据库中添加数据
// 这里的 db 就是第二步中的 db 对象,// transaction api 的第一个参数是数据库名称,第二个参数是操作类型let newItem = { id: 1, name: '徐嘻嘻', age: 3, sex: 'female'};let transaction = db.transaction('dataBaseName', "readwrite");// 找到对应的存储对象let objectStore = transaction.objectStore('person');// 添加到数据对象中, 传入JAVAscript对象objectStore.add(newItem);
- 新建操作是在新建了一个 事务( IDBTransaction 对象)的前提下完成的,传入的数据不需要做任何转换,可以无缝传入 JavaScript 对象 。
- 修改数据库中的数据
// 这里的 db 就是第二步中的 db 对象,// 新建事务let transaction = db.transaction('dataBaseName', "readwrite");// 新数据主体let newRecord = { id: 1, name: '徐嘎嘎', age: 5, sex: 'male'};// 打开已经存储的数据对象let objectStore = transaction.objectStore('person');// 获取存储的对应键的存储对象, 传入主键 id,值为 1 let objectStoreRequest = objectStore.get(1);// 获取成功后替换当前数据objectStoreRequest.onsuccess = function(event) { // 数据 var record = objectStoreRequest.result; // 遍历替换 for (let key in newRecord) { if (typeof record[key] != 'undefined' || key !== 'id') { record[key] = newRecord[key]; } } // 更新数据库存储数据objectStore.put(record);};
推荐阅读
-
-
爆笑小学妹|脸型可能更耐看,爆笑GIF:姑娘你如果把美颜去掉
-
【金十数据|最新:中企320吨蔬菜正运往俄罗斯!中俄推进2000亿美元贸易目标】
-
蒙古国正式启动3万只羊捐赠程序|蒙古国正式启动3万只羊捐赠程序 届时将如何送达?
-
环球网|白宫前高官爆料:特朗普飓风过后不先想着救灾,而想着把波多黎各卖掉!
-
一才说球妻子颜值羡煞旁人,拥7000万豪宅,遭吐槽:拿钱不干事,国脚郜林
-
『国际鲜闻』回击特朗普 美州长联合推出重启各州经济计划,神鸟编译丨
-
花木兰|《花木兰》抢走《八佰》的日冠,口碑有所好转,最终票房将破4亿
-
腾讯|冒充老干妈公司员工 诈骗腾讯上千万广告费 两人被抓
-
#防晒#防晒霜直接涂脸上?难怪容易被晒黑,怎么用都没效果
-
-
天天小花农1|10元一个球,按照3步处理,花开硕大娇艳,捡漏“残次”朱顶红
-
搞笑段子趣图|太不可思议了!,GIF搞笑趣图:什么情况?怎么就充上电了
-
-
-
董思槿风波后现身健身房,真实身材曝光,交往两年的男友已分手
-
『广东男篮』继赵睿之后,广东宏远男篮再出顶薪,这个人就是近日进步飞快的他
-
海南|天猫双11海南成交额24.1亿元!哪些人最能买?这份最全购买榜单来了→
-
岛主历史观|与闺蜜共侍一夫,杜月笙四姨太姚玉兰:撮合孟小冬与杜月笙
-
推迟大选■特朗普发推文后短短几小时就改口:我不想推迟大选