你不知道的JSON.stringify神操

一、前言

在我们面试过程中,面试官经常会提及到深浅拷贝的问题。想必大多数小伙伴会说到JSON.parse(JSON.stringify(obj))。正好今天我就和大家好好唠一唠这个JSON.stringify

二、概念

JSON.stringify对于我们不陌生,一般用来处理序列化(深拷贝)。就是把我们的对象转换JSON字符串,此方法确实很方便在我们的工作中,但是,这个方法也会有一些弊端,只是我们不怎么遇到。

js 复制代码
let obj = {
    name: 'iyongbao'
}

console.log(JSON.stringify(obj)); // {"name":"iyongbao"}

三、弊端

1. 对函数不友好

如果我们的对象属性是一个函数,那么在序列化的时候该属性丢失

js 复制代码
let obj = {
    name: 'iyongbao',
    foo: function () {
        console.log(`${ this.name }是一个小菜鸟!`)
    }
}

console.log(JSON.stringify(obj)); // {"name":"iyongbao"}

2. 对undefined不友好

如果对象的属性值是undefined,转换后会丢失

js 复制代码
let obj = {
    name: undefined
}

console.log(JSON.stringify(obj)); // {}

3. 对正则表达式不友好

如果对象的属性是一个正则表达式,转换后就会变成一个空的Object

js 复制代码
let obj = {
    name: 'iyongbao',
    zoo: /^i/ig,
    foo: function () {
        console.log(`${ this.name }是一个小菜鸟!`)
    }
}

console.log(JSON.stringify(obj)); // {"name":"iyongbao","zoo":{}}

4. 数组对象

如果是一个数组对象,以上的情况也会发生。

js 复制代码
let arr = [
    {
        name: undefined
    }
]

console.log(JSON.stringify(arr)); // [{}]

四、JSON.stringify拓展

说完了JSON.stringify不足,下面我们来说一下你可能没有接触过的其他特性,希望看完会对你有所帮助。

1. 接收一个数组(过滤)

其实JSON.stringify第二参数,可能我们不经常用到。我们可以传入一个数组,值就是对应我们对象key,我称之为过滤。

js 复制代码
let obj = {
    name: 'iyongbao',
    age: 25,
    hobby: ['JavaScript', 'Vue']
}

let res = JSON.stringify(obj, ['name']);

console.log(res); // {"name":"iyongbao"}

2. 接收一个函数

第二个参数也可以是一个函数,也是类似过滤效果

js 复制代码
let obj = {
    name: 'iyongbao',
    age: 25,
    hobby: ['JavaScript', 'Vue']
}

let res = JSON.stringify(obj, (key, value) => {
    if (key === 'age') return undefined;
    return value;
});

console.log(res); // {"name":"iyongbao","hobby":["JavaScript","Vue"]}

3. 缩进

第三个参数可以接收一个数字,表示缩进多少字符

js 复制代码
let obj = {
    name: 'iyongbao',
    age: 25,
    hobby: ['JavaScript', 'Vue']
}

let res = JSON.stringify(obj, null, 2);

console.log(res);

4. 自身toJSON方法

对象可以有一个自身的toJSON属性,是一个返回值方法,用来输出我们自定义的数据样式。

js 复制代码
let obj = {
    name: 'iyongbao',
    age: 25,
    toJSON: function () {
        return {
            message: `${ this.name }的年龄为${ this.age }`
        }
    }
}

let res = JSON.stringify(obj);

console.log(res); // {"message":"iyongbao的年龄为25"}

五、总结

好了,今天就和大家分享到这吧。一般如果真涉及到深拷贝,我还是首选自己封装一个方法或者是使用第三方插件库来做深拷贝,这样最保险,避免不必要的麻烦。

相关推荐
菜鸟一枚在这1 小时前
深入解析设计模式之单例模式
开发语言·javascript·单例模式
ObjectX前端实验室2 小时前
个人网站开发记录-引流公众号 & 谷歌分析 & 谷歌广告 & GTM
前端·程序员·开源
CL_IN2 小时前
企业数据集成:实现高效调拨出库自动化
java·前端·自动化
浪九天3 小时前
Vue 不同大版本与 Node.js 版本匹配的详细参数
前端·vue.js·node.js
qianmoQ3 小时前
第五章:工程化实践 - 第三节 - Tailwind CSS 大型项目最佳实践
前端·css
C#Thread4 小时前
C#上位机--流程控制(IF语句)
开发语言·javascript·ecmascript
椰果uu4 小时前
前端八股万文总结——JS+ES6
前端·javascript·es6
微wx笑4 小时前
chrome扩展程序如何实现国际化
前端·chrome
~废弃回忆 �༄4 小时前
CSS中伪类选择器
前端·javascript·css·css中伪类选择器
CUIYD_19894 小时前
Chrome 浏览器(版本号49之后)‌解决跨域问题
前端·chrome