JavaScript深入理解JSON.stringify

2023-09-15 15:00:00

 🎬 岸边的风:个人主页

 🔥 个人专栏 :《 VUE 》 《 javaScript 》

⛺️ 生活的理想,就是为了理想的生活 !

在这里插入图片描述

目录

引言

1. JSON.stringify() 属性

replacer

space

toJSON

2. 应用场景

数据传输

数据存储

日志记录

数据展示

3. 完整优雅的实现

4. 注意事项

循环引用

特殊类型

性能优化

总结


引言

在 JavaScript 中,JSON.stringify() 是一个内置函数,用于将 JavaScript 对象转换为 JSON 字符串。JSON(JavaScript Object Notation)是一种轻量级的数据交换格式,广泛用于前后端数据传输和存储。本文将详细介绍 JSON.stringify() 的属性、应用场景,并提供一个完整而优雅的实现,处理循环引用、特殊类型(如日期和正则表达式)以及性能相关的问题。同时,我们还将讨论注意事项和相关引用资料。

1. JSON.stringify() 属性

JSON.stringify() 函数具有以下属性:

replacer

replacer 是一个可选的参数,它可以是一个函数或一个数组。它用于指定需要序列化的对象的属性。当 replacer 是一个函数时,它将被应用于对象的每个属性,可以用来过滤、替换或转换属性的值。当 replacer 是一个数组时,只有数组中包含的属性才会被序列化。

示例:

const obj = {
  name: 'John',
  age: 25,
  address: {
    city: 'New York',
    country: 'USA'
  }
};

const jsonString = JSON.stringify(obj, ['name', 'age']);
console.log(jsonString);
// 输出: {"name":"John","age":25}

space

space 是一个可选的参数,用于控制生成的 JSON 字符串的缩进和格式化。它可以是一个数字表示缩进的空格数,或者是一个字符串表示缩进的字符串。如果 space 是一个非负整数,则每一级缩进使用指定数量的空格;如果 space 是一个字符串,则使用该字符串作为缩进符号。

示例:

const obj = { name: 'John', age: 25 };

const jsonString = JSON.stringify(obj, null, 2);
console.log(jsonString);
// 输出:
// {
//   "name": "John",
//   "age": 25
// }

toJSON

如果要序列化的对象具有 toJSON() 方法,那么该方法将被调用,以便返回可序列化的值。toJSON() 方法可以在对象中定义,用于自定义对象在序列化过程中的行为。

示例:

const obj = {
  name: 'John',
  age: 25,
  toJSON: function() {
    return {
      fullName: this.name,
      yearsOld: this.age
    };
  }
};

const jsonString = JSON.stringify(obj);
console.log(jsonString);
// 输出: {"fullName":"John","yearsOld":25}

2. 应用场景

JSON.stringify() 在以下场景中非常有用:

数据传输

当需要将 JavaScript 对象转换为字符串,以便在网络中传输给后端或其他系统时,可以使用 JSON.stringify() 进行序列化。

const obj = { name: 'John', age: 25 };

const jsonString = JSON.stringify(obj);
console.log(jsonString);
// 输出: {"name":"John","age":25}

数据存储

如果需要将 JavaScript 对象保存到本地存储(如浏览器的 LocalStorage 或数据库),可以使用 JSON.stringify() 将对象转换为 JSON 字符串后进行存储。

const obj = { name: 'John', age: 25 };

const jsonString = JSON.stringify(obj);
localStorage.setItem('user', jsonString);

日志记录

在记录日志时,可以将 JavaScript 对象转换为 JSON 字符串,并将其作为日志消息的一部分。

const obj = { name: 'John', age: 25 };

const logMessage = `User info: ${JSON.stringify(obj)}`;
console.log(logMessage);

数据展示

将 JavaScript 对象转换为 JSON 字符串后,可以方便地在前端页面中展示、渲染或打印。

const obj = { name: 'John', age: 25 };

const jsonString = JSON.stringify(obj);
document.getElementById('user-info').textContent = jsonString;

3. 完整优雅的实现

下面是一个完整且优雅的 JSON.stringify() 实现,它考虑了处理循环引用、日期和正则表达式等特殊类型,并尽量保持了性能优化。

function stringify(obj) {
  const seen = new WeakSet(); // 用于检测循环引用
  const typeMap = {
    '[object Date]': 'Date',
    '[object RegExp]': 'RegExp',
  };

  function isObject(value) {
    return typeof value === 'object' && value !== null;
  }

  function handleSpecialTypes(value) {
    if (value instanceof Date) {
      return { type: 'Date', value: value.toISOString() };
    } else if (value instanceof RegExp) {
      return { type: 'RegExp', value: value.toString() };
    }
    return value;
  }

  function replacer(key, value) {
    if (seen.has(value)) {
      throw new TypeError('Converting circular structure to JSON');
    }

    if (isObject(value)) {
      seen.add(value);
    }

    value = handleSpecialTypes(value);

    return value;
  }

  function stringifyHelper(obj) {
    if (isObject(obj)) {
      if (Array.isArray(obj)) {
        return '[' + obj.map((item) => stringifyHelper(item)).join(',') + ']';
      } else {
        const properties = Object.keys(obj)
          .map((key) => `"${key}":${stringifyHelper(obj[key])}`)
          .join(',');
        return `{${properties}}`;
      }
    } else {
      return JSON.stringify(obj, replacer);
    }
  }

  return stringifyHelper(obj);
}

此实现使用了递归和一些辅助函数来处理不同的数据类型。它会检查循环引用并抛出错误,处理特殊类型(如日期和正则表达式),并使用递归进行深度优先遍历。

请注意,此实现仅为简化示例,对于更复杂的场景可能需要进行更多的处理和优化。建议在实际使用中参考第三方库或更全面的文档和资源。

4. 注意事项

在使用

JSON.stringify() 时,需要注意以下事项:

循环引用

如果要序列化的对象存在循环引用,即对象之间相互引用,会导致无限递归的情况。为了避免死循环,可以使用 WeakSet 或其他方式来检测循环引用,并在检测到循环引用时抛出错误或采取其他处理方式。

特殊类型

特殊类型(如日期和正则表达式)需要进行适当的处理,以确保正确的序列化和反序列化。

性能优化

JSON.stringify() 可能会在处理大型对象或嵌套层次较深的对象时产生性能问题。为了提高性能,可以考虑使用更高效的算法或采用其他优化策略。

在实际应用中,了解 JSON.stringify() 的属性、应用场景和实现原理非常重要。通过掌握如何正确使用和实现 JSON.stringify(),我们可以更好地处理和操作 JSON 数据,提高前端开发效率和数据交互的稳定性。

总结

本文详细介绍了 JSON.stringify() 的属性、应用场景,并提供了一个完整而优雅的实现,处理了循环引用、特殊类型(如日期和正则表达式)以及性能优化。我们还讨论了注意事项和相关的参考资料。通过深入了解和熟练掌握 JSON.stringify(),我们可以更好地处理和操作 JSON 数据,提高前端开发的质量和效率。

记住,JSON.stringify() 是处理 JSON 数据的强大工具,但在特殊情况下需要特别小心,确保正确处理特殊类型和避免循环引用的问题。

更多推荐

现场总线学习

文章目录1.现场总线现状2.数据编码2.1数字数据的数字编码2.2数字数据的模拟编码3.通信方式!!!4.局域网及其拓扑结构5.工业总线协议6.为什么要在can协议的控制器和bus总线之间,连接一个can收发器?7.那其他协议也需要这种收发器么?8.OSI网络协议体系1.现场总线现状2.数据编码2.1数字数据的数字编码

PyTorch实战:实现Cifar10彩色图片分类

目录前言一、Cifar10数据集classtorch.utils.data.Datasettorch.utils.data.DataLoader二、定义神经网络普通神经网络:定义损失函数和优化器训练网络-NetCPU训练模型准确率​编辑GPU训练训练网络-LeNet模型准确率点关注,防走丢,如有纰漏之处,请留言指教,非

【智能家居-大模型】构建未来,聆思大模型智能家居交互解决方案正式发布

LISTENAI近日,国内11家大模型陆续通过《生成式人工智能服务管理暂行办法》备案,多家大模型产品已正式开放,激发了新一轮大模型热潮。大模型在自然语言理解方面的巨大突破,实现了认知智能的技术跃迁,带来了时代的智慧涌现,更是为智能家居行业带来了重大的发展契机。依托通用认知大模型,聆思率先上线了面向智能家居领域的大模型人

BEiT: BERT Pre-Training of Image Transformers 论文笔记

BEiT:BERTPre-TrainingofImageTransformers论文笔记论文名称:BEiT:BERTPre-TrainingofImageTransformers论文地址:2106.08254]BEiT:BERTPre-TrainingofImageTransformers(arxiv.org)代码地址

Self-paced Multi-grained Cross-modal Interaction Modeling for Referring Expression Comprehension论文阅读

Self-pacedMulti-grainedCross-modalInteractionModelingforReferringExpressionComprehension论文阅读笔记一、Abstract二、引言三、相关工作A、指代表达式理解B、TransformerC、自定进度学习四、方法A、动机和框架总览B、多

论文阅读《ESSWC2018:Modeling Relational Data with Graph Convolutional Networks》

论文链接本文就是大名鼎鼎的R-GCN,由于是比较早的工作了,在这篇博客中我们只讨论它的链接预测部分和模型结构部分。工作简介本文引入了关系图卷积网络(R-GCNs),并将其应用于两个标准的知识库完成任务:链接预测和实体分类。通过使用GNN编码器模型来丰富用于链接预测的因子分解模型DistMult,可以显著改善链接预测的结

JavaScript中的`async`和`await`关键字的作用

聚沙成塔·每天进步一点点⭐专栏简介⭐async关键字⭐await关键字3.错误处理⭐写在最后⭐专栏简介前端入门之旅:探索Web开发的奇妙世界记得点击上方或者右侧链接订阅本专栏哦几何带你启航前端之旅欢迎来到前端入门之旅!这个专栏是为那些对Web开发感兴趣、刚刚踏入前端领域的朋友们量身打造的。无论你是完全的新手还是有一些基

Leetcode 504.七进制数

给定一个整数num,将其转化为7进制,并以字符串形式输出。示例1:输入:num=100输出:"202"示例2:输入:num=-7输出:"-10"我的答案:一、信息1.目的实现十进制向其他进制的转换。2.原理:公式3.一段运算过程4.根据后面的一段话不难看出根据作者的意思就是用栈的数据结构来存储每次辗转相除后的余数然后输

页面设计都有哪些好用的工具推荐?

对于设计师来说,方便的页面设计工具和稳定的页面设计灵感也同样重要。在今天的信息爆炸中,很容易找到页面设计工具,网上搜索有很多建议,但找到合适的页面设计工具并不那么简单。本文推荐不容错过的9款页面设计工具即时设计即时设计是一款免费的在线UI设计工具,无系统限制,浏览器打开即可使用,更有丰富的在线素材、云端字体、交互动画等

leetcode面试题0808有重复字符串的排列组合

描述输入一个长度为n字符串,打印出该字符串中字符的所有排列,你可以以任意顺序返回这个字符串数组。例如输入字符串ABC,则输出由字符A,B,C所能排列出来的所有字符串ABC,ACB,BAC,BCA,CBA和CAB。数据范围:n<10要求:空间复杂度O(n!),时间复杂度O(n!)输入描述:输入一个字符串,长度不超过10,

不再使用步长卷积或池化:针对低分辨率图像和小物体的新的CNN构建块

摘要https://arxiv.org/pdf/2208.03641.pdf卷积神经网络(CNN)在许多计算机视觉任务中取得了巨大的成功,例如图像分类和目标检测。然而,它们的性能在低分辨率图像或小目标等更困难的图像任务中迅速下降。在本文中,我们指出,这根源于现有CNN体系结构中常见的缺陷设计,即使用步长卷积和/或池化层

热文推荐