你不容错过的JavaScript高级语法(JSON细析, 浏览器数据存储)

简介: 你不容错过的JavaScript高级语法(JSON细析, 浏览器数据存储)

下面一篇文章将介绍JSON, 浏览器数据存储。


JSON


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作为存储格式。


JSON存储类型


JSON的顶层支持三种类型的值:


  • 简单值:数字(Number)、字符串(String,不支持单引号)、布尔类型(Boolean)、null类型。


123


  • 对象值:由key、value组成,key是字符串类型,并且必须添加双引号,值可以是简单值、对象值、数组值。


{
  "name": "zh",
  "age": 22
}


  • 数组值:数组的值可以是简单值、对象值、数组值。


[
      "abc",
      123,
      {
        "name": "zh"
      }
    ]


JSON序列化


某些情况下我们希望将JavaScript中的复杂类型转化成JSON格式的字符串,这样方便对其进行处理:


将一个对象保存到localStorage中。但是如果我们直接存放一个对象,这个对象会被转化成 [object Object] 格式的字符串,并不是我们想要的结果。


JSON序列化方法


遇到上面的情况,我们就需要使用到JSON的方法进行转化。


在ES5中引用了JSON全局对象,该对象有两个常用的方法:


  • stringify方法:将JavaScript类型转成对应的JSON字符串。


  • parse方法:解析JSON字符串,转回对应的JavaScript类型。


const obj = {
      name: "zh",
      age: 22,
    }
    // 将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)


stringify


其中stringify可以传入第二个参数。


  • 主要是用于定制转化的json字符串。


  • 如果是一个 replacer 函数,可以定制传入对象中的值。


const obj = {
      name: 'zh',
      age: 22,
      friends: ["llm", "zhllm"]
    }
    console.log(JSON.stringify(obj,(key, value) => {
      if(key === "name") {
      // value为当前key的值
        return "llm"
      }else if(key === "age") {
        return 21
      }
      // 其他情况返回原本值
      return value
    })) // {"name":"llm","age":21,"friends":["llm","zhllm"]}


  • 如果指定是数组,则将指定传入的对象那些属性需要转化,其他的忽略。


const obj = {
      name: 'zh',
      age: 22,
      friends: ["llm", "zhllm"]
    }
    console.log(JSON.stringify(obj, ["name", "age"])) //{"name":"zh","age":22}


并且stringify还可以传入第三个参数:


  • 用于格式化数据。


  • 如果传入的是Number类型,那么表示每个key都要空number个空格。格式化数据。


const jsonString = JSON.stringify(obj, null, 2)
    console.log(jsonString)


网络异常,图片无法展示
|


  • 如果传入的是String类型,那么表示每个key前面都会被添加的string代替。格式化数据。


const jsonString = JSON.stringify(obj, null, "---")
    console.log(jsonString)


网络异常,图片无法展示
|


如果说传入的对象内部有toJSON方法,那么stringify方法直接返回toJSON方法返回的值。


const obj = {
      name: 'zh',
      age: 22,
      friends: ["llm", "zhllm"],
      toJSON: function() {
        return "Llm love zh."
      }
    }
    const jsonString4 = JSON.stringify(obj)
    console.log(jsonString4) //"Llm love zh."


parse


parse方法也能传递第二个参数。为一个函数,用来定制转化后的值。


const JSONString = `{
      "name": "zh",
      "age": 22,
      "friends": [
        "llm",
        "zhllm"
      ]
    }`
    const info = JSON.parse(JSONString, (key, value) => {
      if (key === "age") {
        return value - 1
      }
      return value
    })
    console.log(info) //{ name: 'zh', age: 21, friends: [ 'llm', 'zhllm' ] }


使用JSON序列化深拷贝


另外我们生成的新对象和之前的对象并不是同一个对象。相当于是进行了一次深拷贝。


const obj = {
      name: 'zh',
      age: 22,
      friends: ["llm", "zhllm"],
      person: {
        name: 'pp'
      },
      foo: function() {
        console.log("foo function")
      }
    }
    const obj1 = JSON.parse(JSON.stringify(obj))
    console.log(obj1.person === obj.person) // false


注意:这种方法它对函数是无能为力的。因为stringify并不会对函数进行处理。


浏览器数据存储


Storage


WebStorage主要提供了一种机制,可以让浏览器提供一种比cookie更直观的key、value存储方式:


  • localStorage:本地存储,提供的是一种永久性的存储方法,在关闭掉网页重新打开时,存储的内容依然保留。


  • sessionStorage:会话存储,提供的是本次会话的存储,在关闭掉会话时,存储的内容会被清除。 localStorage和sessionStorage的区别:


  • 关闭网页后重新打开,localStorage会保留,而sessionStorage会被删除。


  • 在页面内实现跳转,localStorage会保留,sessionStorage也会保留。


  • 在页面外实现跳转(打开新的网页),localStorage会保留,sessionStorage不会被保留。 Storage常见的方法和属性:


  • Storage.length:只读属性


  • 返回一个整数,表示存储在Storage对象中的数据项数量。


  • Storage.key():该方法接受一个数值n(从0开始)作为参数,返回存储中的第n个key名称。


  • Storage.getItem():该方法接受一个key作为参数,并且返回key对应的value。


  • Storage.setItem():该方法接受一个key和value,并且将会把key和value添加到存储中。如果key存储,则更新其对应的值。


  • Storage.removeItem():该方法接受一个key作为参数,并把该key从存储中删除。


  • Storage.clear():该方法的作用是清空存储中的所有key。 下面来对localStorage和sessionStorage进行封装并对其进行JSON序列化,方便以后使用:


class Cache {
      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 Cache()
    const sessionCache = new Cache(false)
    export {
      localCache,
      sessionCache
    }


IndexedDB


什么是IndexedDB呢?


我们能看到DB这个词,就说明它其实是一种数据库(Database),通常情况下在服务器端比较常见。在实际的开发中,大量的数据都是存储在数据库的,客户端主要是请求这些数据并且展示。


有时候我们可能会存储一些简单的数据到本地(浏览器中),比如token、用户名、密码、用户信息等,比较少存储大量的数据。那么如果确实有大量的数据需要存储,这个时候可以选择使用IndexedDB。


  • IndexedDB是一种底层的API,用于在客户端存储大量的结构化数据。


  • 它是一种事务型数据库系统,是一种基于JavaScript面向对象数据库,有点类似于NoSQL(非关系型数据库)。


  • IndexDB本身就是基于事务的,我们只需要指定数据库模式,打开与数据库的连接,然后检索和更新一系列事务即可。


IndexedDB的连接数据库


第一步:打开indexedDB的某一个数据库:


  • 通过indexDB.open(数据库名称, 数据库版本)方法。


  • 如果数据库不存在,那么会创建这个数据。


  • 如果数据库已经存在,那么会打开这个数据库。


// 打开数据(和数据库建立连接)
    const dbRequest = indexedDB.open("zhDB", 2)


第二步:通过监听回调得到数据库连接结果:


  • 数据库的open方法会得到一个IDBOpenDBRequest类型


  • 我们可以通过下面的三个回调来确定结果:


  • onerror:当数据库连接失败时。


  • onsuccess:当数据库连接成功时回调。我们可以通过onsuccess回调的event获取到db对象:event.target.result。


  • onupgradeneeded:当数据库第一次打开或者version发生变化并且高于之前版本时回调。通常我们在这里会创建具体的存储对象:db.createObjectStore(存储对象名称, { keypath: 存储的主键 })


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
      // 创建一些存储对象
      db.createObjectStore("users", { keyPath: "id" })
    }


IndexedDB的数据库操作


我们对数据库的操作要通过事务对象来完成:


  • 第一步:通过db获取对应存储的事务 db.transaction(存储名称, 可写操作)。


  • 第二步:通过事务获取对应的存储对象 transaction.objectStore(存储名称)。


const transaction = db.transaction("users", "readwrite")
    const store = transaction.objectStore("users")


接下来我们就可以进行增删改查操作了:


  • 增。通过add方法。


const request = store.add(user)
    // 单条数据存储成功后的回调
    request.onsuccess = function() {
        console.log(`${user.name}插入成功`)
    }
    // 全部数据存储成功的回调。
    transaction.oncomplete = function() {
      console.log("添加操作全部完成")
    }


  • 查。


  • 方式一:get方法,只能传入主键, 根据主键查询。


  • 方式二:openCursor方法,然后调用成功时的回调,然后查询全部数据,然后if过滤数据。


// 方式一
    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.result
      if (cursor) {
        if (cursor.key === 101) {
          console.log(cursor.key, cursor.value)
        } else {
          cursor.continue()
        }
      } else {
        console.log("查询完成")
      }
    }


  • 改。还是先查找,然后调用该记录的update方法。将修改后的数据作为参数传入。


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 = "curry"
          cursor.update(value)
        } else {
          cursor.continue()
        }
      } else {
        console.log("查询完成")
      }
    }


  • 删。还是先查找,然后调用该记录的delete方法。


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("查询完成")
      }
    }


详细知识请访问mdn下面来看一下案例


<button>新增</button>
  <button>查询</button>
  <button>删除</button>
  <button>修改</button>
  <script src="./index.js"></script>


// 打开数据(和数据库建立连接)
    const dbRequest = indexedDB.open("zhDB", 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, "zh", 18),
      new User(101, "llm", 40),
      new User(102, "zh-llm", 30),
    ]
    // 获取btns, 监听点击
    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("添加操作全部完成")
            }
            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 = "curry"
                  cursor.update(value)
                } else {
                  cursor.continue()
                }
              } else {
                console.log("查询完成")
              }
            }
            break
        }
      }
    }


网络异常,图片无法展示
|



相关文章
|
4月前
|
Web App开发 JavaScript 前端开发
添加浮动按钮点击滚动到网页底部的纯JavaScript演示代码 IE9、11,Maxthon 1.6.7,Firefox30、31,360极速浏览器7.5.3.308下测试正常
添加浮动按钮点击滚动到网页底部的纯JavaScript演示代码 IE9、11,Maxthon 1.6.7,Firefox30、31,360极速浏览器7.5.3.308下测试正常
|
2月前
|
移动开发 JavaScript 前端开发
一些处理浏览器兼容性问题的JavaScript库
这些库在处理浏览器兼容性问题方面都有着各自的特点和优势,可以根据具体的需求和项目情况选择合适的库来使用,从而提高代码的兼容性和稳定性,为用户提供更好的体验。同时,随着浏览器技术的不断发展,还需要持续关注和学习新的兼容性解决方案。
116 48
|
2月前
|
JSON 移动开发 JavaScript
在浏览器执行js脚本的两种方式
【10月更文挑战第20天】本文介绍了在浏览器中执行HTTP请求的两种方式:`fetch`和`XMLHttpRequest`。`fetch`支持GET和POST请求,返回Promise对象,可以方便地处理异步操作。`XMLHttpRequest`则通过回调函数处理请求结果,适用于需要兼容旧浏览器的场景。文中还提供了具体的代码示例。
在浏览器执行js脚本的两种方式
|
2月前
|
JavaScript 前端开发 数据处理
模板字符串和普通字符串在浏览器和 Node.js 中的性能表现是否一致?
综上所述,模板字符串和普通字符串在浏览器和 Node.js 中的性能表现既有相似之处,也有不同之处。在实际应用中,需要根据具体的场景和性能需求来选择使用哪种字符串处理方式,以达到最佳的性能和开发效率。
|
2月前
|
算法 开发者
Moment.js库是如何处理不同浏览器的时间戳格式差异的?
总的来说,Moment.js 通过一系列的技术手段和策略,有效地处理了不同浏览器的时间戳格式差异,为开发者提供了一个稳定、可靠且易于使用的时间处理工具。
56 1
|
2月前
|
Web App开发 JavaScript 前端开发
使用 Chrome 浏览器的内存分析工具来检测 JavaScript 中的内存泄漏
【10月更文挑战第25天】利用 Chrome 浏览器的内存分析工具,可以较为准确地检测 JavaScript 中的内存泄漏问题,并帮助我们找出潜在的泄漏点,以便采取相应的解决措施。
393 9
|
2月前
|
机器学习/深度学习 自然语言处理 前端开发
前端神经网络入门:Brain.js - 详细介绍和对比不同的实现 - CNN、RNN、DNN、FFNN -无需准备环境打开浏览器即可测试运行-支持WebGPU加速
本文介绍了如何使用 JavaScript 神经网络库 **Brain.js** 实现不同类型的神经网络,包括前馈神经网络(FFNN)、深度神经网络(DNN)和循环神经网络(RNN)。通过简单的示例和代码,帮助前端开发者快速入门并理解神经网络的基本概念。文章还对比了各类神经网络的特点和适用场景,并简要介绍了卷积神经网络(CNN)的替代方案。
240 1
|
2月前
|
JSON 关系型数据库 MySQL
MySQL JSON数据存储结构与操作
通过本文的介绍,我们了解了MySQL中JSON数据类型的基本操作、常用JSON函数、以及如何通过索引和优化来提高查询性能。JSON数据类型为存储和操作结构化数据提供了灵活性和便利性,在现代数据库应用中具有广泛的应用前景。希望本文对您在MySQL中使用JSON数据类型有所帮助。
236 0
|
3月前
|
机器学习/深度学习 自然语言处理 前端开发
前端大模型入门:Transformer.js 和 Xenova-引领浏览器端的机器学习变革
除了调用API接口使用Transformer技术,你是否想过在浏览器中运行大模型?Xenova团队推出的Transformer.js,基于JavaScript,让开发者能在浏览器中本地加载和执行预训练模型,无需依赖服务器。该库利用WebAssembly和WebGPU技术,大幅提升性能,尤其适合隐私保护、离线应用和低延迟交互场景。无论是NLP任务还是实时文本生成,Transformer.js都提供了强大支持,成为构建浏览器AI应用的核心工具。
697 1
|
3月前
|
Web App开发 JavaScript 前端开发
JavaScript Window - 浏览器对象模型
JavaScript Window - 浏览器对象模型
32 2