你不知道的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 小时前
常见 CSS 选择器用法
前端·css·学习·html·前端开发·css选择器
悦涵仙子1 小时前
CSS中的变量应用——:root,Sass变量,JavaScript中使用Sass变量
javascript·css·sass
衣乌安、1 小时前
【CSS】居中样式
前端·css·css3
兔老大的胡萝卜1 小时前
ppk谈JavaScript,悟透JavaScript,精通CSS高级Web,JavaScript DOM编程艺术,高性能JavaScript pdf
前端·javascript
低代码布道师1 小时前
CSS的三个重点
前端·css
耶啵奶膘2 小时前
uniapp-是否删除
linux·前端·uni-app
王哈哈^_^4 小时前
【数据集】【YOLO】【目标检测】交通事故识别数据集 8939 张,YOLO道路事故目标检测实战训练教程!
前端·人工智能·深度学习·yolo·目标检测·计算机视觉·pyqt
cs_dn_Jie4 小时前
钉钉 H5 微应用 手机端调试
前端·javascript·vue.js·vue·钉钉
开心工作室_kaic5 小时前
ssm068海鲜自助餐厅系统+vue(论文+源码)_kaic
前端·javascript·vue.js
有梦想的刺儿5 小时前
webWorker基本用法
前端·javascript·vue.js