当前位置: 首页 > news >正文

玩转IndexedDB,比localStorage、cookie还要强大的网页端本地缓存


随着浏览器的功能不断增强,越来越多的网站开始考虑,将大量数据储存在客户端,这样可以减少从服务器获取数据,直接从本地获取数据。

现有的浏览器数据储存方案,都不适合储存大量数据:Cookie 的大小不超过 4KB,且每次请求都会发送回服务器;LocalStorage 在 2.5MB 到 10MB 之间(各家浏览器不同),而且不提供搜索功能,不能建立自定义的索引。所以,需要一种新的解决方案,这就是 IndexedDB 诞生的背景。

通俗地说,IndexedDB 就是浏览器提供的本地数据库,它可以被网页脚本创建和操作。IndexedDB 允许储存大量数据,提供查找接口,还能建立索引。这些都是 LocalStorage 所不具备的。就数据库类型而言,IndexedDB 不属于关系型数据库(不支持 SQL 查询语句),更接近 NoSQL 数据库。

对比cookielocalStoragesessionStorageindexedDB
存储大小4kb5M5M很多于 250MB,甚至没有上限
与服务器端通讯每次都会携带在HTTP头中,若是使用cookie保存过多数据会带来性能问题仅在客户端(即浏览器)中保存,不参与和服务器的通讯
生命周期通常由服务器生成,可设置失效时间。若是在浏览器端生成Cookie,默认是关闭浏览器后失效除非被清除,不然永久保存仅在当前会话下有效,关闭页面或浏览器后被清除除非被清除,不然永久保存
使用场景判断用户是否登陆存储一些内容稳定的资源。好比图片内容丰富的电商网站会用它来存储 Base64 格式的图片字符串存储一些当前会话的信息,好比微博的 sessionStorage就主要是存储你本次会话的浏览足迹和 localStorage 用途相似:
1.  存储量会更大
2. localStorage使用简单字符串键值对在本地存储数据,而indexedDB能够存储任意类型的值(适合键值对较多的数据,若是使用 localStorage 存储每次都要写入,写出须要字符串化和对象化)
复制代码


目前,Chrome 27+、Firefox 21+、Opera 15+和IE 10+支持这个API,但是Safari完全不支持。

下面的代码用来检查浏览器是否支持这个API。

if("indexedDB" in window) {// 支持
} else {// 不支持
}

IndexedDB 具有以下特点。

(1)键值对储存。 IndexedDB 内部采用对象仓库(object store)存放数据。所有类型的数据都可以直接存入,包括 JavaScript 对象。对象仓库中,数据以“键值对”的形式保存,每一个数据记录都有对应的主键,主键是独一无二的,不能有重复,否则会抛出一个错误。

(2)异步。 IndexedDB 操作时不会锁死浏览器,用户依然可以进行其他操作,这与 LocalStorage 形成对比,后者的操作是同步的。异步设计是为了防止大量数据的读写,拖慢网页的表现。

(3)支持事务。 IndexedDB 支持事务(transaction),这意味着一系列操作步骤之中,只要有一步失败,整个事务就都取消,数据库回滚到事务发生之前的状态,不存在只改写一部分数据的情况。

(4)同源限制。 IndexedDB 受到同源限制,每一个数据库对应创建它的域名。网页只能访问自身域名下的数据库,而不能访问跨域的数据库。

(5)储存空间大。 IndexedDB 的储存空间比 LocalStorage 大得多,一般来说不少于 250MB,甚至没有上限。

(6)支持二进制储存。 IndexedDB 不仅可以储存字符串,还可以储存二进制数据(ArrayBuffer 对象和 Blob 对象)。

IndexedDB 是一个比较复杂的 API,涉及不少概念。它把不同的实体,抽象成一个个对象接口。学习这个 API,就是学习它的各种对象接口。

  • 数据库:IDBDatabase 对象
  • 对象仓库:IDBObjectStore 对象
  • 索引: IDBIndex 对象
  • 事务: IDBTransaction 对象
  • 操作请求:IDBRequest 对象
  • 指针: IDBCursor 对象
  • 主键集合:IDBKeyRange 对象

下面是一些主要的概念。

(1)数据库

数据库是一系列相关数据的容器。每个域名(严格的说,是协议 + 域名 + 端口)都可以新建任意多个数据库。

IndexedDB 数据库有版本的概念。同一个时刻,只能有一个版本的数据库存在。如果要修改数据库结构(新增或删除表、索引或者主键),只能通过升级数据库版本完成。

(2)对象仓库

每个数据库包含若干个对象仓库(object store)。它类似于关系型数据库的表格。

(3)数据记录

对象仓库保存的是数据记录。每条记录类似于关系型数据库的行,但是只有主键和数据体两部分。主键用来建立默认的索引,必须是不同的,否则会报错。主键可以是数据记录里面的一个属性,也可以指定为一个递增的整数编号。

{ id: 1, value: '对应的值' }

上面的对象中,id属性可以当作主键。

数据体可以是任意数据类型,不限于对象。

(4)索引

为了加速数据的检索,可以在对象仓库里面,为不同的属性建立索引。

(5)事务

数据记录的读写和删改,都要通过事务完成。事务对象提供errorabortcomplete三个事件,用来监听操作结果。


用例代码 

<template><div><el-input v-model.trim="databaseName" :placeholder="`请输入数据库名称`" /><el-input v-model.trim="tableName" :placeholder="`请输入表名称`" /><el-input v-model.trim="index" :placeholder="`请输入字段名`" /><el-button type="primary" @click="btn1()">创建数据库</el-button><hr><el-input v-model.trim="value" :placeholder="`请输入值`" /><el-button type="success" @click="btn2()">添加数据</el-button><hr><el-input v-model.trim="keyPathValue" :placeholder="`请输入主键值`" /><el-input v-model.trim="newValue" :placeholder="`请输入修改值`" /><el-button type="warning" @click="btn3()">修改数据</el-button><hr><el-button type="info" @click="btn4()">读取数据</el-button><template v-if="tableData.length"><el-button type="danger" @click="btn5">删除全部</el-button><el-table :data="tableData"><el-table-column :prop="keyPath" :label="keyPath" /><el-table-column :prop="index" :label="index" /><el-table-column label="操作"><template slot-scope="scope"><el-button size="mini" type="danger" @click.stop="btn6(scope.row[keyPath])">删除</el-button></template></el-table-column></el-table></template></div>
</template>
<script>
export default {data() {return {databaseName: '',tableName: '',keyPath: 'id',keyPathValue: '',index: '',value: '',newValue: '',tableData: [],}},methods: {// 用例----------------------------------------// 创建btn1() {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;let keyPath = this.keyPath;let indexs = [[this.index, this.index, { unique: false }],];//如需定义多格字段,就多几个数组// 创建表this.creatDatabaseTable({databaseName, version,tableName,//定义表名keyPath,//定义主键indexs,// 定义索引字段});},// 添加btn2() {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;let keyPath = this.keyPath;let index = this.index;this.addData({databaseName, version, tableName,data: {[keyPath]: '********'.replace(/\*/g, () => Math.round(Math.random() * 15).toString(16)),//随机id[index]: this.value,},onsuccess: d => { console.log(`onsuccess`, d); },onerror: d => { console.log(`onerror`, d); },})},// 修改btn3() {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;let keyPath = this.keyPath;let index = this.index;this.updateData({databaseName, version, tableName,data: {[keyPath]: this.keyPathValue,[index]: this.newValue,}});this.btn4();//刷新数据},// 读取btn4() {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;this.readData({databaseName, version, tableName,onsuccess: ({ data }) => { this.tableData = data },})},// 删除全部btn5() {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;this.delAllData({ databaseName, version, tableName, });this.btn4();//刷新数据},// 删除btn6(id) {let databaseName = this.databaseName;let version = this.version;let tableName = this.tableName;this.delData({databaseName, version, tableName,data: id,//需要删除的数据主键onsuccess: d => {this.btn4();//刷新数据},onerror: d => { console.log(`onerror`, d); },})},// indexedDB----------------------------------------// 1、创建or打开客户端数据库createDatabase({ databaseName, version = 1, onupgradeneeded, onsuccess, onerror } = {}) {let request = window.indexedDB.open(databaseName, version);request.onupgradeneeded = onupgradeneeded;request.onsuccess = onsuccess;request.onerror = onerror;},getDatabase(obj) { return this.createDatabase(obj); },//获取数据库// 2、创建表creatDatabaseTable({ databaseName, version, tableName, keyPath, indexs, onupgradeneeded, onsuccess, onerror } = {}) {this.getDatabase({databaseName, version,onupgradeneeded: d => {let database = d.target.result;if (!database.objectStoreNames.contains(tableName)) {//createObjectStore只能在onupgradeneeded里面执行let objectStore = database.createObjectStore(tableName, { keyPath });(indexs || []).forEach(v => objectStore.createIndex(...v));onupgradeneeded && onupgradeneeded({ event: d, objectStore });}},onsuccess,onerror,})},// 3、添加数据or修改数据or删除数据addData({ databaseName, version, tableName, data, onsuccess, onerror, triggerName = 'add' } = {}) {this.getDatabase({databaseName, version,onsuccess: d => {let database = d.target.result;let objectStore = database.transaction(tableName, 'readwrite').objectStore(tableName);let request_objectStore = objectStore.get(data[objectStore.keyPath]);request_objectStore.onsuccess = event => {event.target.result && (triggerName = 'put');//如果已经存在该主键数据,就变成修改let request = objectStore[triggerName](data);request.onsuccess = onsuccess;request.onerror = onerror;};},})},// 4、修改数据updateData(obj) { this.addData({ ...obj, triggerName: 'put' }) },// 5、删除数据delData(obj) { this.addData({ ...obj, triggerName: 'delete' }) },delAllData({ databaseName, version, tableName } = {}) {this.getDatabase({databaseName, version,onsuccess: d => {let objectStore = d.target.result.transaction(tableName, 'readwrite').objectStore(tableName);objectStore.clear();},})},// 6、读取数据readData({ databaseName, version, tableName, onsuccess } = {}) {this.getDatabase({databaseName, version,onsuccess: d => {let database = d.target.result;if (database.objectStoreNames.contains(tableName)) {let objectStore = database.transaction(tableName).objectStore(tableName);let data = [];objectStore.openCursor().onsuccess = event => {let cursor = event.target.result;if (cursor) {data.push(cursor.value); cursor.continue();} else {onsuccess && onsuccess({ event, data });// console.log('没有更多数据了!'); }};} else onsuccess && onsuccess({ msg: '表格不存在!', data: [] });},})},// ----------------------------------------}
};
</script>

http://www.lryc.cn/news/123543.html

相关文章:

  • RedisDesktopManager连不上redis问题解决(小白版)
  • 蓝帽杯 取证2022
  • MyBatis and or使用列表控制or条件
  • C语言刷题训练【第11天】
  • 正则表达式的使用
  • PHP 求解两字符串所有公共子序列及最长公共子序列 支持多字节字符串
  • linux内核bitmap之setbit汇编实现
  • Golang设计模式
  • leetcode151. 反转字符串中的单词
  • 【BASH】回顾与知识点梳理(十七)
  • 时序预测-Informer简介
  • 2023牛客第七场补题报告C F L M
  • Android使用kotlin+协程+room数据库的简单应用
  • Kubernetes pod调度约束[亲和性 污点] 生命阶段 排障手段
  • Matlab实现模拟退火算法(附上多个完整源码)
  • 前后端分离------后端创建笔记(03)前后端对接(上)
  • stable diffusion安装包和超火使用文档及提示词,数字人网址
  • 训练营:贪心篇
  • 四、Dubbo扩展点加载机制
  • [保研/考研机试] KY103 2的幂次方 上海交通大学复试上机题 C++实现
  • 时序预测 | MATLAB实现基于BP神经网络的时间序列预测-递归预测未来(多指标评价)
  • 组合模式(C++)
  • git上传问题记录
  • 通过动态IP解决网络数据采集问题
  • 可重入锁,不可重入锁,死锁的多种情况,以及产生的原因,如何解决,synchronized采用的锁策略(渣女圣经)自适应的底层,锁清除,锁粗化,CAS的部分应用
  • JSON.parse()和JSON.stringify()用法
  • Android 并发编程--阻塞队列和线程池
  • Playwright快速上手-1
  • PPT颜色又丑又乱怎么办?
  • python计算相关系数R