你不知道的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"}

五、总结

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

相关推荐
旺旺大力包16 分钟前
【 Git 】git 的安装和使用
前端·笔记·git
PP东21 分钟前
ES学习class类用法(十一)
javascript·学习
海威的技术博客26 分钟前
JS中的原型与原型链
开发语言·javascript·原型模式
雪落满地香32 分钟前
前端:改变鼠标点击物体的颜色
前端
余生H1 小时前
前端Python应用指南(二)深入Flask:理解Flask的应用结构与模块化设计
前端·后端·python·flask·全栈
outstanding木槿1 小时前
JS中for循环里的ajax请求不数据
前端·javascript·react.js·ajax
酥饼~1 小时前
html固定头和第一列简单例子
前端·javascript·html
一只不会编程的猫1 小时前
高德地图自定义折线矢量图形
前端·vue.js·vue
所以经济危机就是没有新技术拉动增长了1 小时前
二、javascript的进阶知识
开发语言·javascript·ecmascript
m0_748250931 小时前
html 通用错误页面
前端·html