当前位置:首页 > 文章列表 > 文章 > 前端 > JSON.stringify的作用及使用场景详解

JSON.stringify的作用及使用场景详解

2025-08-30 15:42:33 0浏览 收藏

`JSON.stringify()` 是 JavaScript 中一个至关重要的方法,它能将 JavaScript 对象或值转换成 JSON 字符串,实现数据的序列化,从而方便数据在网络传输和本地存储。在 Web 开发中,无论是前端向后端发送数据,还是将配置信息存储在 `localStorage` 中,`JSON.stringify()` 都扮演着核心角色。本文将深入解析 `JSON.stringify()` 的作用与常见使用场景,例如利用 `fetch` API 发送 `POST` 请求,以及在 `localStorage` 中存储复杂对象。同时,本文还将详细介绍使用该方法处理特殊数据类型(如函数、`undefined`、循环引用等)时的注意事项,并探讨如何利用 `replacer` 和 `space` 参数优化 JSON 字符串的输出,提升代码的可读性和灵活性,从而帮助开发者更好地掌握这一实用工具。

JSON.stringify的核心作用是将JavaScript对象或值转换为JSON字符串,便于数据传输(如fetch发送POST请求)和存储(如localStorage);2. 处理特殊类型时需注意:函数、undefined、Symbol会被忽略,数组中对应值变null,循环引用会抛错,Date转ISO字符串但反序列化仍为字符串,BigInt和Map/Set需手动处理;3. 通过replacer参数可过滤属性或自定义转换逻辑(如隐藏敏感信息、处理BigInt),space参数可格式化输出(数字为空格数,字符串为缩进符),提升可读性与灵活性。

js 中 JSON.stringify 方法作用 js 中 JSON.stringify 方法的使用场景

JSON.stringify 方法在 JavaScript 里,简单来说,就是把一个 JavaScript 值(比如对象、数组、字符串、数字等)转换成一个 JSON 格式的字符串。它的主要作用就是为了数据的序列化,让数据能够方便地在网络上传输,或者存储到本地,因为它把复杂的数据结构变成了一个统一的、文本化的格式。

js 中 JSON.stringify 方法作用 js 中 JSON.stringify 方法的使用场景

JSON.stringify 方法的核心作用是将 JavaScript 对象或值转换为 JSON 字符串。这在 Web 开发中几乎无处不在,尤其当你需要将前端的数据发送到后端服务器,或者将一些配置、状态保存到本地存储(如 localStoragesessionStorage)时,它都是不可或缺的工具。它确保了数据在不同系统、不同语言之间传递时的格式一致性,因为 JSON 本身就是一种轻量级的数据交换格式。

JSON.stringify 在实际开发中常见的使用场景有哪些?

我个人觉得,JSON.stringify 在日常开发里简直是万金油般的存在,用得最多的,毫无疑问就是数据传输和数据存储。

js 中 JSON.stringify 方法作用 js 中 JSON.stringify 方法的使用场景

首先说数据传输。想象一下,你在前端页面里收集了一大堆用户输入的信息,比如一个表单对象,里面有姓名、年龄、地址,甚至还有一些嵌套的选项。你要把这些数据发给后端服务器,如果直接发一个 JavaScript 对象过去,那后端可能根本不认识,或者解析起来会非常麻烦。这时候,JSON.stringify 就派上用场了,它能把这个复杂的 JavaScript 对象变成一个标准的 JSON 字符串,后端用任何语言(Python、Java、Node.js等)都能轻松解析。比如,fetch API 或者 XMLHttpRequest 发送 POST 请求时,body 通常就需要一个字符串化的数据。

const userData = {
  name: '张三',
  age: 30,
  address: {
    city: '北京',
    street: '某某街'
  },
  hobbies: ['阅读', '编程']
};

// 发送到服务器前,先序列化
fetch('/api/users', {
  method: 'POST',
  headers: {
    'Content-Type': 'application/json'
  },
  body: JSON.stringify(userData) // 这里就用到了!
})
.then(response => response.json())
.then(data => console.log('Success:', data))
.catch(error => console.error('Error:', error));

再就是数据存储。浏览器提供的 localStoragesessionStorage 只能存储字符串。如果你想把一个 JavaScript 对象或者数组存进去,直接存是会变成 [object Object] 这样的字符串的,根本不是你想要的数据。所以,你得先用 JSON.stringify 把它变成 JSON 字符串,存进去;取出来的时候,再用 JSON.parse 把它变回原来的 JavaScript 对象。这简直是前端持久化数据最基础的手段了。

js 中 JSON.stringify 方法作用 js 中 JSON.stringify 方法的使用场景
const appSettings = {
  theme: 'dark',
  notifications: true,
  lastLogin: new Date()
};

// 存储前序列化
localStorage.setItem('appSettings', JSON.stringify(appSettings));

// 取出后反序列化
const storedSettings = JSON.parse(localStorage.getItem('appSettings'));
console.log(storedSettings.theme); // 'dark'

还有一些不那么显眼,但同样重要的场景,比如深拷贝。虽然不是最推荐的深拷贝方式(因为它有一些限制),但对于只包含基本数据类型、数组和普通对象的简单数据结构来说,JSON.parse(JSON.stringify(obj)) 确实是一种快速实现深拷贝的“黑科技”。不过,这个方法不能处理函数、undefinedDate 对象、RegExp 对象、MapSet 等特殊类型,所以用的时候得清楚它的局限性。

最后,调试时也挺好用。有时候你想看看一个复杂对象里面到底有什么,直接 console.log(obj) 可能输出得不够清晰,或者对象引用关系让你摸不着头脑。这时候,console.log(JSON.stringify(obj, null, 2)) 就能把对象格式化成易读的 JSON 字符串,方便你检查数据结构和内容。

JSON.stringify 处理特殊数据类型(如函数、undefined、循环引用)时有哪些注意事项?

在使用 JSON.stringify 的时候,有一些数据类型它处理起来会比较“个性”,甚至会直接报错,这些点是必须要注意的,不然很容易踩坑。

我遇到的最常见的问题就是函数和 undefined。当 JSON.stringify 遇到对象中的属性值是函数、undefined 或 Symbol 值时,它会直接忽略这些属性,不把它们包含在最终的 JSON 字符串里。如果这些值是数组的元素,那么它们会被转换成 null。这听起来有点奇怪,但这是 JSON 标准的规定。所以,如果你期望序列化一个包含函数或 undefined 的对象,并且希望这些信息也保留下来,那 JSON.stringify 就不适合直接用,你可能需要自己手动处理,或者用其他序列化库。

const complexObj = {
  name: '测试',
  age: undefined, // undefined
  greet: function() { console.log('Hello'); }, // 函数
  symbolKey: Symbol('id'), // Symbol
  data: [1, undefined, 3, function() {}] // 数组中的 undefined 和函数
};

console.log(JSON.stringify(complexObj));
// 输出: {"name":"测试","data":[1,null,3,null]}
// 注意:age、greet、symbolKey 都没了,数组里的 undefined 和函数变成了 null

另一个大坑是循环引用。如果你的对象结构里存在循环引用,比如 a 引用了 b,同时 b 又引用了 a,那么 JSON.stringify 在尝试序列化时会陷入无限循环,最终抛出一个 TypeError: Converting circular structure to JSON 的错误。这是非常致命的,因为这会直接导致你的程序崩溃。解决这个问题通常需要你在序列化之前手动解除循环引用,或者使用一些专门处理循环引用的库。我通常会在遇到这种错误时,第一时间检查数据结构,看看是不是不小心创建了循环引用。

const objA = {};
const objB = {};
objA.b = objB;
objB.a = objA; // 循环引用!

try {
  JSON.stringify(objA); // 这里会抛出 TypeError
} catch (e) {
  console.error("捕获到错误:", e.message);
  // 输出: 捕获到错误: Converting circular structure to JSON
}

还有一些值得一提的:

  • Date 对象Date 对象会被转换为 ISO 格式的字符串,比如 "2023-10-27T08:00:00.000Z"。这通常是符合预期的,但在反序列化(JSON.parse)时,它仍然是一个字符串,需要你手动将其转换回 Date 对象。
  • BigIntBigInt 类型的值在 JSON.stringify 时会直接抛出 TypeError。JSON 标准本身不支持 BigInt。如果你需要序列化 BigInt,通常需要先将其转换为 String,然后在反序列化时再转回 BigInt
  • MapSet:这些新的数据结构也不会被直接序列化成它们的原有结构,它们会被转换成空对象 {}。如果你需要序列化它们,通常需要手动将它们转换为数组或其他 JSON 支持的类型。

了解这些“怪癖”非常重要,它能帮助你避免很多不必要的调试时间和错误。

如何利用 replacer 和 space 参数优化 JSON.stringify 的输出?

JSON.stringify 提供了两个可选参数:replacerspace,它们能让你对序列化过程有更精细的控制,这在很多场景下都非常有用。

先说 space 参数,这个参数主要是为了美化输出。如果你给它传入一个数字(0-10),它就会用这个数字表示的空格数来缩进 JSON 字符串,让它更易读。如果传入一个字符串(比如 '\t'),它就会用这个字符串作为缩进字符。这在调试时特别有用,因为格式化的 JSON 字符串比一行紧凑的字符串看着舒服多了,一眼就能看出数据结构。

const data = {
  name: '示例数据',
  details: {
    id: 123,
    status: 'active'
  },
  items: ['itemA', 'itemB']
};

// 不带 space 参数,输出紧凑
console.log('紧凑输出:', JSON.stringify(data));
// 输出: {"name":"示例数据","details":{"id":123,"status":"active"},"items":["itemA","itemB"]}

// space 为 2,缩进 2 个空格
console.log('缩进 2 个空格:', JSON.stringify(data, null, 2));
/*
输出:
{
  "name": "示例数据",
  "details": {
    "id": 123,
    "status": "active"
  },
  "items": [
    "itemA",
    "itemB"
  ]
}
*/

// space 为 '\t',使用制表符缩进
console.log('使用制表符缩进:', JSON.stringify(data, null, '\t'));
/*
输出:
{
    "name": "示例数据",
    "details": {
        "id": 123,
        "status": "active"
    },
    "items": [
        "itemA",
        "itemB"
    ]
}
*/

再来说说 replacer 参数,这个参数就强大多了,它允许你自定义序列化过程中的值转换replacer 可以是一个数组,也可以是一个函数。

replacer 是一个数组时,这个数组里应该包含你想要保留的属性名(字符串)。JSON.stringify 只会序列化这些指定属性及其子属性。这在你想从一个大对象中提取部分信息进行序列化时非常方便,可以有效减少传输的数据量。

const userProfile = {
  id: 'user123',
  name: '李四',
  email: 'lisi@example.com',
  passwordHash: 'some_hash_value', // 不希望被序列化
  lastLogin: new Date()
};

// 只序列化 id, name, email
const publicProfile = JSON.stringify(userProfile, ['id', 'name', 'email']);
console.log('部分序列化:', publicProfile);
// 输出: {"id":"user123","name":"李四","email":"lisi@example.com"}

replacer 是一个函数时,它会为对象中的每个属性(包括对象本身)调用一次。这个函数接收两个参数:key(属性名)和 value(属性值)。你可以在这个函数里根据 keyvalue 返回不同的值,从而实现更复杂的转换逻辑。

  • 如果你返回 value,则保持原样。
  • 如果你返回 undefined,则该属性会被忽略(不会出现在最终的 JSON 字符串中)。
  • 如果你返回其他值,则该属性会被替换为返回的值。

这个功能非常灵活,比如你可以用来过滤敏感信息、转换数据类型、或者处理那些 JSON.stringify 默认不支持的类型(如 BigIntDate 对象,如果你想以特定格式保存它们的话)。

const product = {
  name: '超级电脑',
  price: 9999.99,
  stock: 100,
  description: '性能强劲,值得拥有',
  secretKey: 'abc-123-xyz', // 敏感信息
  manufactureDate: new Date(),
  bigNumber: 12345678901234567890n // BigInt
};

function customReplacer(key, value) {
  // 过滤掉敏感信息
  if (key === 'secretKey') {
    return undefined; // 忽略这个属性
  }
  // 将 BigInt 转换为字符串
  if (typeof value === 'bigint') {
    return value.toString();
  }
  // 将 Date 对象转换为 YYYY-MM-DD 格式的字符串
  if (value instanceof Date) {
    return value.toISOString().split('T')[0];
  }
  // 其他值保持不变
  return value;
}

const serializedProduct = JSON.stringify(product, customReplacer, 2);
console.log('自定义序列化:', serializedProduct);
/*
输出:
{
  "name": "超级电脑",
  "price": 9999.99,
  "stock": 100,
  "description": "性能强劲,值得拥有",
  "manufactureDate": "2023-10-27", // 日期格式化了
  "bigNumber": "12345678901234567890" // BigInt 变字符串了
}
*/

通过灵活运用 replacerspace,我们不仅能控制 JSON 字符串的内容,还能控制它的可读性,这对于构建健壮和高效的 Web 应用来说,是不可或缺的技能。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于文章的相关知识,也可关注golang学习网公众号。

Golang反射调用函数与参数传递技巧Golang反射调用函数与参数传递技巧
上一篇
Golang反射调用函数与参数传递技巧
PHP数组添加元素的几种方法
下一篇
PHP数组添加元素的几种方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    511次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    498次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 千音漫语:智能声音创作助手,AI配音、音视频翻译一站搞定!
    千音漫语
    千音漫语,北京熠声科技倾力打造的智能声音创作助手,提供AI配音、音视频翻译、语音识别、声音克隆等强大功能,助力有声书制作、视频创作、教育培训等领域,官网:https://qianyin123.com
    518次使用
  • MiniWork:智能高效AI工具平台,一站式工作学习效率解决方案
    MiniWork
    MiniWork是一款智能高效的AI工具平台,专为提升工作与学习效率而设计。整合文本处理、图像生成、营销策划及运营管理等多元AI工具,提供精准智能解决方案,让复杂工作简单高效。
    483次使用
  • NoCode (nocode.cn):零代码构建应用、网站、管理系统,降低开发门槛
    NoCode
    NoCode (nocode.cn)是领先的无代码开发平台,通过拖放、AI对话等简单操作,助您快速创建各类应用、网站与管理系统。无需编程知识,轻松实现个人生活、商业经营、企业管理多场景需求,大幅降低开发门槛,高效低成本。
    508次使用
  • 达医智影:阿里巴巴达摩院医疗AI影像早筛平台,CT一扫多筛癌症急慢病
    达医智影
    达医智影,阿里巴巴达摩院医疗AI创新力作。全球率先利用平扫CT实现“一扫多筛”,仅一次CT扫描即可高效识别多种癌症、急症及慢病,为疾病早期发现提供智能、精准的AI影像早筛解决方案。
    530次使用
  • 智慧芽Eureka:更懂技术创新的AI Agent平台,助力研发效率飞跃
    智慧芽Eureka
    智慧芽Eureka,专为技术创新打造的AI Agent平台。深度理解专利、研发、生物医药、材料、科创等复杂场景,通过专家级AI Agent精准执行任务,智能化工作流解放70%生产力,让您专注核心创新。
    511次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码