JSON-数据存储


theme: channing-cyan

跟随coderwhy学习

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.小程序的app.json

image.png

3.JSON基本语法

  • JSON的顶层支持三种类型的值:
    • 简单值:数字(Number)、字符串(String,不支持单引号)、布尔类型(Boolean)、null类型;
    • 对象值:由key、value组成,key是字符串类型,并且必须添加双引号,值可以是简单值、对象值、数组值;
    • 数组值:数组的值可以是简单值、对象值、数组值;

image.png

4.JSON序列化

  • 某些情况下我们希望将JavaScript中的复杂类型转化成JSON格式的字符串,这样方便对其进行处理:
    • 比如我们希望将一个对象保存到localStorage中;
    • 但是如果我们直接存放一个对象,这个对象会被转化成 [object Object] 格式的字符串,并不是我们想要的结果;

image.png

image.png

5.JSON序列化方法

  • 在ES5中引用了JSON全局对象,该对象有两个常用的方法:
    • stringify方法:将JavaScript类型转成对应的JSON字符串;
    • parse方法:解析JSON字符串,转回对应的JavaScript类型;
  • 那么上面的代码我们可以通过如下的方法来使用:

    ```js // 将obj转成JSON格式的字符串 const objString = JSON.stringify(obj) // 将对象数据存储localStorage localStorage.setItem("obj", objString)

    const jsonString = localStorage.getItem("obj") // 将JSON格式的字符串转回对象 const info = JSON.parse(jsonString) console.log(info) ```

    6.Stringify的参数replace

  • JSON.stringify() 方法将一个 JavaScript 对象或值转换为 JSON 字符串:

    • 如果指定了一个 replacer 函数,则可以选择性地替换值;
    • 如果指定的 replacer 是数组,则可选择性地仅包含数组指定的属性; ```js // 需求: 将上面对象转成JSON字符串 // 1.直接转化 const jsonString1 = JSON.stringify(obj) // {"name":"mint","friends":{"name":"pika"},"bobbies":["篮球","足球"]} console.log(jsonString1)

      // 2.stringify第二个参数replacer // 2.1 传入数组:设定哪些是需要转换 const jsonString2 = JSON.stringify(obj, ["name", "friends"]) // {"name":"mint","friends":{"name":"pika"}} console.log(jsonString2)

      // 2.2 传入回调函数 const jsonString3 = JSON.stringify(obj, (key, value) => { if (key === 'age') { return value +1 } return value }) console.log(jsonString3) ```

    7.Stringify的参数space

  • 当然,它还可以跟上第三个参数space js // 3.stringify第三参数 space const jsonString4 = JSON.stringify(obj, null, '---') console.log(jsonString4);

  • 如果对象本身包含toJSON方法,那么会直接使用toJSON方法的结果: ```js const obj = { "name": "mint", "friends": { "name": "pika" }, "bobbies": ["篮球", "足球"], toJSON: function () { return 'mint' } }

const objString5 = JSON.stringify(obj) console.log(objString5)//mint ```

8..parse方法

  • JSON.parse() 方法用来解析JSON字符串,构造由字符串描述的JavaScript值或对象。
    • 提供可选的 reviver 函数用以在返回之前对所得到的对象执行变换(操作)。

9.使用JSON序列化深拷贝

  • 另外我们生成的新对象和之前的对象并不是同一个对象:

    • 相当于是进行了一次深拷贝; ```js const objString = JSON.stringify(obj) const info = JSON.parse(objString) // { name: 'mint', age: 18, friend: { name: 'pika' } } console.log(info)

    console.log(info === obj) // false info.name.name = 'andy' console.log(obj.friend.name) // pika ```

  • 注意:这种方法它对函数是无能为力的
    • 创建出来的info中是没有foo函数的,这是因为stringify并不会对函数进行处理;
    • 我们后续会讲解如何编写深拷贝的工具函数,那么这样就可以对函数的拷贝进行处理了;

10.认识Storage

  • WebStorage主要提供了一种机制,可以让浏览器提供一种比cookie更直观的key、value存储方式:
    • localStorage:本地存储,提供的是一种永久性的存储方法,在关闭掉网页重新打开时,存储的内容依然保留;
    • sessionStorage:会话存储,提供的是本次会话的存储,在关闭掉会话时,存储的内容会被清除;

image.png

11.localStorage和sessionStorage的区别

  • 我们会发现localStorage和sessionStorage看起来非常的相似。
  • 那么它们有什么区别呢?

    • 验证一:关闭网页后重新打开,localStorage会保留,而sessionStorage会被删除;
    • 验证二:在页面内实现跳转,localStorage会保留,sessionStorage也会保留;
    • 验证三:在页面外实现跳转(打开新的网页),localStorage会保留,sessionStorage不会被保留;

    12.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;

13.封装Storage

  • 在开发中,为了让我们对Storage使用更加方便,我们可以对其进行一些封装: ```js class myCache { 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(key) { return this.storage.key(index) }

    length() { return this.storage.length } }

const localCache = new myCache(true) const sessionCache = new myCache(false)

export { localCache, sessionCache } ```

14.认识IndexedDB

  • 什么是IndexedDB呢?
    • 我们能看到DB这个词,就说明它其实是一种数据库(Database),通常情况下在服务器端比较常见;
    • 在实际的开发中,大量的数据都是存储在数据库的,客户端主要是请求这些数据并且展示;
    • 有时候我们可能会存储一些简单的数据到本地(浏览器中),比如token、用户名、密码、用户信息等,比较少存储大量的数 据;
    • 那么如果确实有大量的数据需要存储,这个时候可以选择使用IndexedDB;
  • IndexedDB是一种底层的API,用于在客户端存储大量的结构化数据。
    • 它是一种事务型数据库系统,是一种基于JavaScript面向对象数据库,有点类似于NoSQL(非关系型数据库);
    • IndexDB本身就是基于事务的,我们只需要指定数据库模式,打开与数据库的连接,然后检索和更新一系列事务即可;

image.png

15.IndexDB的连接数据库

  • 第一步:打开indexDB的某一个数据库;
    • 通过indexDB.open(数据库名称, 数据库版本)方法;
    • 如果数据库不存在,那么会创建这个数据;
    • 如果数据库已经存在,那么会打开这个数据库;
  • 第二步:通过监听回调得到数据库连接结果;
    • 数据库的open方法会得到一个IDBOpenDBRequest类型
    • 我们可以通过下面的三个回调来确定结果:
      • onerror:当数据库连接失败时;
      • onsuccess:当数据库连接成功时回调;
      • onupgradeneeded:当数据库的version发生变化并且高于之前版本时回调;
        • 通常我们在这里会创建具体的存储对象:db.createObjectStore(存储对象名称, { keypath: 存储的主键 })
    • 我们可以通过onsuccess回调的event获取到db对象:event.target.result

16.IndexedDB的数据库操作

  • 我们对数据库的操作要通过事务对象来完成:
    • 第一步:通过db获取对应存储的事务 db.transaction(存储名称, 可写操作);
      • 第二步:通过事务获取对应的存储对象 transaction.objectStore(存储名称);
  • 接下来我们就可以进行增删改查操作了:
    • 新增数据 store.add
    • 查询数据
      • 方式一:store.get(key)
      • 方式二:通过 store.openCursor 拿到游标对象
        • 在request.onsuccess中获取cursor:event.target.result
        • 获取对应的key:cursor.key;
        • 获取对应的value:cursor.value;
        • 可以通过cursor.continue来继续执行;
    • 修改数据 cursor.update(value)
    • 删除数据 cursor.delete()

17.IndexedDB操作的代码

```js // 打开数据(和数据库建立连接) const dbRequest = indexedDB.open('mint', 3) dbRequest.onerror = function (err) { console.log('打开数据库失败~') }

let db = null dbRequest.onsuccess = function (event) { db = event.target.result }

// 第一次打开/或者版本发生升级 dbRequest.onupgradeneeded = function (event) { const db = event.target.result

console.log(db)

// 创建一些储存对象
db.createObjectStore('users', {keyPath: 'id'})

}

class User { constructor(id, name, age) { this.id = id this.name = name this.age = age } }

const users = [ new User(100, 'mint', 18), new User(101, 'pika', 21), new User(102, 'andy', 19) ]

// 获取btns,监听点击 const btns = document.querySelectorAll('button') for (let i = 0; i < btns.length; i++) { btns[i].onclick = function () { const transaction = db.transaction('users', 'readwrite') console.log(transaction) 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('添加操作全部完成')
            }
            break
        case 1:
            console.log('点击了查询')
            // 1. 查询方式一(知道主键,更具主键查询)
            // const request = store.get(102)
            // request.onsuccess = function (event) {
            //     console.log(event.target.result)
            // }

            // 2. 查询方式二:
            const request = store.openCursor()
            request.onsuccess = function (event) {
                const cursor = event.target.result
                if (cursor) {
                    if (cursor.key === 101) {
                        console.log(cursor.key, cursor.value)
                    } else {
                        cursor.continue()
                    }
                } else {
                    console.log('查询完成')
                }
            }
            break
        case 2:
            console.log('点击了删除')
            const deleteRequest = store.openCursor()
            deleteRequest.onsuccess = function (event) {
                const cursor = event.target.result
                if (cursor) {
                    if (cursor.key === 101) {
                        cursor.delete()
                    } else {
                        cursor.continue()
                    }
                } else {
                    console.log('查询完成')
                }
            }
            break
        case 3:
            console.log('点击了修改')
            const updateRequest = store.openCursor()
            updateRequest.onsuccess = function (event) {
                const cursor = event.target.result
                if (cursor) {
                    if (cursor.key === 101) {
                        const value = cursor.value;
                        value.name = 'pika'
                        cursor.update(value)
                    } else {
                        cursor.continue()
                    }
                } else {
                    console.log('查询完成')
                }
            }
            break
    }
}

} ```

18.认识Cookie

  • Cookie(复数形态Cookies),又称为“小甜饼”。类型为“小型文本文件,某些网站为了辨别用户身份而存储在用户本地终端 (Client Side)上的数据。
    • 浏览器会在特定的情况下携带上cookie来发送请求,我们可以通过cookie来获取一些信息;
  • Cookie总是保存在客户端中,按在客户端中的存储位置,Cookie可以分为内存Cookie和硬盘Cookie。
    • 内存Cookie由浏览器维护,保存在内存中,浏览器关闭时Cookie就会消失,其存在时间是短暂的;
    • 硬盘Cookie保存在硬盘中,有一个过期时间,用户手动清理或者过期时间到时,才会被清理;
  • 如果判断一个cookie是内存cookie还是硬盘cookie呢?
    • 没有设置过期时间,默认情况下cookie是内存cookie,在关闭浏览器时会自动删除;
    • 有设置过期时间,并且过期时间不为0或者负数的cookie,是硬盘cookie,需要手动或者到期时,才会删除;

image.png

19.cookie常见的属性

  • cookie的生命周期:
    • 默认情况下的cookie是内存cookie,也称之为会话cookie,也就是在浏览器关闭时会自动被删除;
    • 我们可以通过设置expires或者max-age来设置过期的时间;
      • expires:设置的是Date.toUTCString(),设置格式是;expires=date-in-GMTString-format;
      • max-age:设置过期的秒钟,;max-age=max-age-in-seconds (例如一年为606024*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

20.客户端设置cookie

  • js直接设置和获取cookie: js console.log(document.cookie);
  • 这个cookie会在会话关闭时被删除掉; js // 设置过期时间就是本地cookie,不设置就是内存cookie doucument.cookie = "name=mint" doucument.cookie = "age=18"
  • 设置cookie,同时设置过期时间(默认单位是秒钟) js document.cookie = "name=mint;max-age=10"
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值