title: Storage
categories: Javascript
tag:
- storage
date: 2021-11-30 11:35:34
认识 Storage
WebStorage 主要提供了一种机制,可以让浏览器提供一种比 cookie 更直观的 key、value 存储方式:
- localStorage:本地存储,提供的是一种永久性的存储方法,在关闭掉网页重新打开时,存储的内容依然保留;
- sessionStorage:会话存储,提供的是本次会话的存储,在关闭掉会话时,存储的内容会被清除;
localStorage 和 sessionStorage 区别
我们会发现 localStorage 和 sessionStorage 看起来非常的相似。
那么它们有什么区别呢?
- 验证一:关闭网页后重新打开,localStorage 会保留,而 sessionStorage 会被删除;
- 验证二:在页面内实现跳转,localStorage 会保留,sessionStorage 也会保留;
- 验证三:在页面外实现跳转(打开新的网页),localStorage 会保留,sessionStorage 不会被保留;
Storage 常见的方法和属性
Storage 有如下的属性和方法:
属性:
- Storage.length:只读属性
- 返回一个整数,表示存储在 Storage 对象中的数据项数量;
方法:
- Storage.key():该方法接受一个数值 n 作为参数,返回存储中的第 n 个 key 名称;
- Storage.getItem():该方法接受一个 key 作为参数,并且返回 key 对应的 value;
- Storage.setItem():该方法接受一个 key 和 value,并且将会把 key 和 value 添加到存储中。
- 如果 key 存储,则更新其对应的值;
- Storage.removeItem():该方法接受一个 key 作为参数,并把该 key 从存储中删除;
- Storage.clear():该方法的作用是清空存储中的所有 key;
封装 Storage

class HyCache {constructor(isLocal = true) {this.storage = isLocal ? localStorage : sessionStorage}setItem(key, value) {if (value) {this.storage.setItem(key, JSON.stringify(value))}}getItem(key) {let value = this.storage.getItem(key)if (value) {value = JSON.parse(value)return value}}removeItem(key) {this.storage.removeItem(key)}clear() {this.storage.clear()}key(index) {return this.storage.key(index)}length() {return this.storage.length}}const localCache = new HyCache()const sessionCache = new HyCache()export { localCache, sessionCache }
认识 IndexedDB
什么是 IndexedDB 呢?
- 我们能看到 DB 这个词,就说明它其实是一种数据库(Database),通常情况下在服务器端比较常见;
- 在实际的开发中,大量的数据都是存储在数据库的,客户端主要是请求这些数据并且展示;
- 有时候我们可能会存储一些简单的数据到本地(浏览器中),比如 token、用户名、密码、用户信息等,比较少存储大量的数据;
- 那么如果确实有大量的数据需要存储,这个时候可以选择使用 IndexedDB;
IndexedDB 是一种底层的 API,用于在客户端存储大量的结构化数据。
- 它是一种事务型数据库系统,是一种基于 JavaScript 面向对象数据库,有点类似于 NoSQL(非关系型数据库);
- IndexDB 本身就是基于事务的,我们只需要指定数据库模式,打开与数据库的连接,然后检索和更新一系列事务即可;
IndexedDB 的连接数据库
第一步:打开 indexDB 的某一个数据库;
- 通过 indexDB.open(数据库名称, 数据库版本)方法;
- 如果数据库不存在,那么会创建这个数据;
- 如果数据库已经存在,那么会打开这个数据库;
第二步:通过监听回调得到数据库连接结果;
- 数据库的 open 方法会得到一个IDBOpenDBRequest类型
- 我们可以通过下面的三个回调来确定结果:
- onerror:当数据库连接失败时;
- onsuccess:当数据库连接成功时回调;
- onupgradeneeded:当数据库的 version 发生变化并且高于之前版本时回调;
- 通常我们在这里会创建具体的存储对象:db.createObjectStore(存储对象名称, { keypath: 存储的主键 })
- 我们可以通过 onsuccess 回调的 event 获取到 db 对象:event.target.result
//why相当于数据库的库const dbRequest = indexedDB.open('why')dbRequest.onerror = function (err) {console.log('打开数据库失败')}let db = nulldbRequest.onsuccess = function (event) {db = event.target.result}// 第一次打开或者版本更新dbRequest.onupgradeneeded = function (event) {const db = event.target.result//创建一些存储对象.// users相当于数据库中的表// keyPath相当于数据库中的primary keydb.createObjectStore('users', { keyPath: 'id' })}

IndexedDB 的数据库操作
新增
我们先以构造函数的方式批量新建对象
class User {constructor(id, name, age) {this.id = idthis.name = namethis.age = age}}const users = [new User(100, 'why', 18),new User(101, 'dh', 22),new User(102, 'gwk', 22),new User(103, 'kk', 30)]
然后点击新增按钮。插入数据

查询
case 1:console.log('点击了查询')//查询方式1:(根据主键查询)const request = store.get(102)request.onsuccess = function (event) {console.log(event.target.result)}break

批量查询
case 1:console.log('点击了查询')const request = store.openCursor()request.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {console.log(cursor.key, cursor.value)cursor.continue()} else {console.log('查询完成')}}break

修改
case 2:console.log('点击了修改')const updateRequest = store.openCursor()updateRequest.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {if (cursor.key === 101) {const value = cursor.valuevalue.name = 'curry'cursor.update(value)console.log('修改成功')} else {cursor.continue()}}}break

删除
case 3:console.log('点击了删除')const deleteRequest = store.openCursor()deleteRequest.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {if (cursor.key === 101) {cursor.delete()console.log('删除成功')} else {cursor.continue()}}}

完整代码
const dbRequest = indexedDB.open('why')dbRequest.onerror = function (err) {console.log('打开数据库失败')}let db = nulldbRequest.onsuccess = function (event) {db = event.target.result}// 第一次打开或者版本更新dbRequest.onupgradeneeded = function (event) {const db = event.target.result//创建一些存储对象.// users相当于数据库中的表// keyPath相当于数据库中的primary keydb.createObjectStore('users', { keyPath: 'id' })}class User {constructor(id, name, age) {this.id = idthis.name = namethis.age = age}}const users = [new User(100, 'why', 18),new User(101, 'dh', 22),new User(102, 'gwk', 22),new User(103, 'kk', 30)]const btns = document.querySelectorAll('button')for (let i = 0; i < btns.length; i++) {btns[i].onclick = function () {const transaction = db.transaction('users', 'readwrite')const store = transaction.objectStore('users')switch (i) {case 0:console.log('点击了新增')for (const user of users) {const request = store.add(user)request.onsuccess = function () {console.log(`插入${user.name}成功`)}}transaction.oncomplete = function () {console.log('添加操作全部完成')}breakcase 1:console.log('点击了查询')//查询(根据主键查询)// const request = store.get(102)// request.onsuccess = function (event) {// console.log(event.target.result)// }const request = store.openCursor()request.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {console.log(cursor.key, cursor.value)cursor.continue()} else {console.log('查询完成')}}breakcase 2:console.log('点击了修改')const updateRequest = store.openCursor()updateRequest.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {if (cursor.key === 101) {const value = cursor.valuevalue.name = 'curry'cursor.update(value)console.log('修改成功')} else {cursor.continue()}}}breakcase 3:console.log('点击了删除')const deleteRequest = store.openCursor()deleteRequest.onsuccess = function (event) {const cursor = event.target.resultif (cursor) {if (cursor.key === 101) {cursor.delete()console.log('删除成功')} else {cursor.continue()}}}}}}
