JSON语法,JSON序列化(stringify,parse),localStorage,sessionStorage,indexedDB(增删改查),Cookie
1. JSON的由来
在目前的开发中,JSON是一种
非常重要的数据格式,它并不是编程语言,而是一种可以在服务器和客户端之间传输的数据格式JSON的全称是JavaScript Object Notation(JavaScript对象符号)
- JSON是由Douglas Crockford构想和设计的一种轻量级资料交换格式,算是JavaScript的一个子集
- 但是虽然JSON被提出来的时候是主要应用JavaScript中,但是目前已经独立于编程语言,可以在各个编程语言中使用
- 很多编程语言都实现了将JSON转成对应模型的方式
其他的传输格式
XML:在早期的网络传输中主要是使用XML来进行数据交换的,但是这种格式在解析、传输等各方面都弱于JSON,所以目前已经很少在被使用了Protobuf:另外一个在网络传输中目前已经越来越多使用的传输格式是protobuf,但是直到2021年的3.x版本才支持JavaScript,所以目前在前端使用的较少
目前JSON被使用的场景也越来越多
- 网络数据的传输JSON数据
- 项目的某些配置文件
- 非关系型数据库(NoSQL)将json作为存储格式
2. JSON基本语法
JSON的顶层支持三种类型的值
简单值:数字(Number)、字符串(String,不支持单引号)、布尔类型(Boolean)、null类型
1
123
对象值:由key、value组成,key是字符串类型,并且必须添加
双引号,值可以是简单值、对象值、数组值1
2
3
4{
"name":"why",
"age": 18
}数组值:数组的值可以是简单值、对象值、数组值
1
2
3
4
5
6
7
8[
123,
"abc",
{
"name":"why",
"age":18
}
]
3. JSON序列化-对象存储存在问题
- 将JavaScript中的复杂类型转化成JSON格式的字符串,这样方便对其进行处理
- 比如将一个对象保存到localStorage中
- 但是如果直接存放一个对象,这个对象会被转化成 [object Object] 格式的字符串,并不是想要的结果
1 | const obj = { |

4. JSON序列化方法
在ES5中引用了JSON全局对象,该对象有两个常用的方法
stringify方法:将JavaScript类型转成对应的JSON字符串parse方法:解析JSON字符串,转回对应的JavaScript类型
那么上面的代码通过如下的方法来使用
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17const obj = {
name: "why",
age: 18
}
// localStorage.setItem("obj",obj);
// 存储
// 转换为字符串
let str = JSON.stringify(obj);
localStorage.setItem("obj",str);
// 获取
str = localStorage.getItem("obj");
// 转换为对象
const getObj = JSON.parse(str);
console.log(getObj);

4.1. Stringify的参数replacer
- JSON.stringify() 方法将一个 JavaScript 对象或值转换为 JSON 字符串
- 如果指定了一个
replacer 函数,则可以选择性地替换值 - 如果指定的
replacer 是数组,则可选择性地仅包含数组指定的属性
- 如果指定了一个
1 | const obj = { |
4.2. Stringify的参数space
它还可以跟上第三个参数space,规定字符串输出时的缩进
- 为数字时,表示缩进空格
- 为字符串时,表示缩进字符串形式
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22const obj = {
name: "why",
age: 18
};
str = JSON.stringify(obj,null,2);
/**
{
"name": "why",
"age": 18
}
*/
console.log(str);
str = JSON.stringify(obj,null,"--");
/**
{
--"name": "why",
--"age": 18
}
*/
console.log(str);如果对象本身包含toJSON方法,那么会直接使用toJSON方法的结果
1
2
3
4
5
6
7
8
9
10
11const obj = {
name: "why",
age: 18,
toJSON: () => {
return "Hello World";
}
};
str = JSON.stringify(obj, null, 2);
// Hello World
console.log(str);
4.3. parse方法
- JSON.parse() 方法用来解析JSON字符串,构造由字符串描述的JavaScript值或对象
- 提供可选的 reviver 函数用以在返回之前对所得到的对象执行变换(操作)
1 | const obj = { |
4.4. 对象的拷贝
对象相互赋值分别包括
- 引入的赋值:指向同一个对象,相互之间会影响
- 对象的浅拷贝:只是浅层的拷贝,内部引入对象时,依然会相互影响
- 对象的深拷贝:两个对象不再有任何关系,不会相互影响
4.4.1. 引用赋值
- 生成的新对象和之前的对象指向的是同一个对象
1 | const obj = { |
4.4.2. 浅拷贝
- 生成的新对象和之前的对象并不是同一个对象
- 但是对象内部中对象的指向仍是指向同一个对象
- obj的friends对象和info的friends对象指向同一个对象
1 | const obj = { |
4.4.3. 使用JSON序列化深拷贝
生成的新对象和之前的对象并不是同一个对象
- 相当于是进行了一次深拷贝
这种方法它对函数是无能为力的
- 创建出来的info中是没有foo函数的,这是因为stringify并不会对函数进行处理
1 | const obj = { |
4.4.4. 自定义深拷贝函数
实现深拷贝
- JSON.parse
- 这种深拷贝的方式其实对于函数、Symbol等是无法处理的
- 并且如果存在对象的循环引用,也会报错
- JSON.parse
自定义深拷贝函数
- 自定义深拷贝的基本功能
- 对Symbol的key进行处理
- 其他数据类型的值进程处理:数组、函数、Symbol、Set、Map
- 对循环引用的处理
obj对象
1 | const s1 = Symbol(); |
深拷贝实现
1 | function isObject(value) { |
5. 认识Storage
- 数据存储在用户浏览器中
- 设置,读取方便,甚至页面刷新不丢失数据
- 容量较大,sessionStorage约5M,localStorage约30M
- 只能存储字符串,可以将对象JSON.stringify()编码后存储
- localStorage:本地存储,提供的是一种永久性的存储方法,在关闭掉网页重新打开时,存储的内容依然保留,同一个浏览器下数据共享,以键值对形式存储
- sessionStorage:会话存储,提供的是本次会话的存储,在关闭掉会话时,存储的内容会被清除,同一个窗口下数据共享,以键值对形式存储
5.1. localStorage和sessionStorage的区别
- 关闭网页后重新打开,localStorage会保留,而sessionStorage会被删除
- 在页面内实现跳转,localStorage会保留,sessionStorage也会保留
- 在页面外实现跳转(打开新的网页),localStorage会保留,sessionStorage不会被保留
5.2. Storage常见的方法和属性
5.2.1. 属性
- Storage.length:只读属性
- 返回一个整数,表示存储在Storage对象中的数据项数量
5.2.2. 方法
- Storage.key(n):该方法接受一个数值n作为参数,返回存储中的第n个key名称
- Storage.getItem(key):该方法接受一个key作为参数,并且返回key对应的value
- Storage.setItem(key, value):该方法接受一个key和value,并且将会把key和value添加到存储中
- 如果key存储,则更新其对应的值
- Storage.removeItem():该方法接受一个key作为参数,并把该key从存储中删除
- Storage.clear():该方法的作用是清空存储中的所有key
5.3. 封装Storage
- 在开发中,为了对Storage使用更加方便,可以对其进行一些封装
1 | class MyCache { |
6. IndexedDB
6.1. 认识IndexedDB
什么是IndexedDB呢?
- 看到DB这个词,就说明它其实是一种数据库(Database),通常情况下在服务器端比较常见
- 在实际的开发中,大量的数据都是存储在数据库的,客户端主要是请求这些数据并且展示
- 有时候可能会存储一些简单的数据到本地(浏览器中),比如token、用户名、密码、用户信息等,比较少存储大量的数据
- 那么如果确实有大量的数据需要存储,可以选择IndexedDB
IndexedDB是一种底层的API,用于在客户端存储大量的结构化数据
- 它是一种事务型数据库系统,是一种基于JavaScript面向对象数据库,有点类似于NoSQL(非关系型数据库)
- IndexDB本身就是基于事务的,只需要指定数据库模式,打开与数据库的连接,然后检索和更新一系列事务即可
6.2. 数据库操作
6.2.1. 连接数据库
第一步:打开indexDB的某一个数据库
通过
indexDB.open(数据库名称, 数据库版本) 方法如果数据库不存在,那么会创建这个数据
如果数据库已经存在,那么会打开这个数据库
1 | // 打开数据库 |
6.2.2. 监听回调
- 第二步:通过监听回调得到数据库连接结果
- 数据库的open方法会得到一个 IDBOpenDBRequest 类型
- 可以通过下面的三个回调来确定结果
- onerror:当数据库连接失败时
- onsuccess:当数据库连接成功时回调
- 可以通过 IDBOpenDBRequest 的 result属性 获取到db对象:dbRequest.result
- onupgradeneeded:当第一次打开数据库,或者数据库的version发生变化并且高于之前版本时回调
- 通常会创建具体的存储对象
- const db = event.target.result
- db.createObjectStore(存储对象名称, { keypath: 存储的主键 })
1 | // 数据库对象 |
6.2.3. 事务对象
IndexedDB 规定,在对新数据库做任何事情之前,需要开始一个事务, 事务中需要指定该事务跨越哪些存储对象 ( objectStore )
对于事务,一个简单的理解就是,一个事务里的操作,要么全部执行成功,要么全部执行失败
IndexedDB 提供了
IDBDatabase.transaction()方法用于开启是一个事务,并且返回一个IDBTransaction对象该方法的原型是
IDBDatabase.transaction(storeName, [mode])storeName:必填,用于指定事务中需要操作到的存储对象,如果只有一个存储对象,可以直接写存储对象名,如果有多个存储对象,那么需要把它们封装成一个列表
mode:可选,用于指定事务的类型,可选的值有:
readonly( 只读,默认值 ) 或者readwrite(读写)
获取
IDBTransaction对象步骤- 第一步:通过 IDBDatabase 获取对应存储的事务
- 第二步:通过事务获取对应的存储对象 transaction.objectStore(存储名称)
1 | // 1.通过db获取对应存储的事务 |
6.2.4. 新增数据
- store.add()
1 | class User { |
6.2.5. 查找数据
单个
- store.get(key)
多个
通过 store.openCursor 拿到游标对象
在request.onsuccess中获取cursor:event.target.result
获取对应的key:cursor.key
获取对应的value:cursor.value
可以通过cursor.continue()来继续执行
1 | // 单个查询 |
6.2.6. 删除数据
- cursor.delete()
1 | // 删 |
6.2.7. 修改数据
- cursor.update(value)
1 | // 改 |
7. cookie
7.1. 认识cookie
- Cooki(复数形态Cookies),又称为“小甜饼”。类型为小型文本文,某些网站为了辨别用户身份而存储在用户本地终端(Client Side)上的数据
- 浏览器会在特定的情况下携带上cookie来发送请求,可以通过cookie来获取一些信息
Cookie总是保存在客户端中,按在客户端中的存储位置,Cookie可以分为内存Cookie和硬盘Cookie
- 内存Cookie由浏览器维护,保存在内存中,浏览器关闭时Cookie就会消失,其存在时间是短暂的
- 硬盘Cookie保存在硬盘中,有一个过期时间,用户手动清理或者过期时间到时,才会被清理
如果判断一个cookie是内存cookie还是硬盘cookie呢?
- 没有设置过期时间,默认情况下cookie是内存cookie,在关闭浏览器时会自动删除
- 有设置过期时间,并且过期时间不为0或者负数的cookie,是硬盘cookie,需要手动或者到期时,才会删除
7.2. cookie代码测试
1 | const Koa = require("koa"); |
/test 设置cookie

/demo 获取cookie

7.3. cookie常见的属性
- cookie的生命周期
- 默认情况下的cookie是内存cookie,也称之为会话cookie,也就是在浏览器关闭时会自动被删除
- 通过设置expires或者max-age来设置过期的时间
- expires:设置的是Date.toUTCString(),设置格式是:expires=date-in-GMTString-format
- max-age:设置过期的秒钟,max-age=max-age-in-seconds (例如一年为
60*60*24*365)
- cookie的作用域:允许cookie发送给哪些URL
- Domain:指定哪些主机可以接受cookie
- 如果不指定,那么默认是 origin,不包括子域名
- 如果指定Domain,则包含子域名。例如,如果设置 Domain=mozilla.org,则 Cookie 也包含在子域名中(如developer.mozilla.org)
- Path:指定主机下哪些路径可以接受cookie,例如,设置 Path=/docs,则以下地址都会匹配
- /docs
- /docs/Web/
- /docs/Web/HTTP
- Domain:指定哪些主机可以接受cookie
7.4. 客户端设置cookie
js直接设置和获取cookie
1
document.cookie
不设置cookie过期时间就是内存cookie,会在会话关闭时被删除掉
1
2document.cookie = "name=hello"
document.cookie = "age=18"设置cookie,同时设置过期时间(默认单位是秒钟)
1
document.cookie = "name=why;max-age=10"
设置cookie过期
1
document.cookie = "name=why;max-age=0"
