寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

本文详细探讨了JavaScript中对象和数组的深拷贝与浅拷贝的区别,包括它们的工作原理、应用场景以及如何实现。通过实例展示了各种拷贝方法的效果,帮助读者深入理解拷贝机制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

(内容同步自小邹的头条号:沪漂程序员的生活史)

对象和数组的拷贝对小邹来说一直都是一个比较模糊的概念,一直有点一知半解,但是在实际工作中又偶尔会涉及到,有时候还会一不小心掉坑里,不知道大家有没有同样的感受,因此,准备对js对象和数组拷贝一探究竟。提到js的对象和数组拷贝,大家一定会想深拷贝和浅拷贝,但是为什么会有深拷贝和浅拷贝呢?下面就让我简单介绍一下为什么拷贝会有深浅之分以及有什么区别?

原因及区别

我们都知道js中有两种数据类型,一种是基本数据类型,一种是引用数据类型,基本数据类型是按值访问的,即在操作基本类型的变量时,是直接修改变量的值,而引用数据类型的值是按引用访问的,什么叫按引用访问的呢?js的引用类型,也叫对象类型,是保存在内存中的,而在js中又无法直接操作内存中的对象,实际上操作的是对象的引用,因此在引用类型变量在进行复制操作时,并不是对对象值的直接复制,而是将对象的引用复制给了另一个变量,实际上变量指向的是同一个内存地址中对象的值,因此只要改变其中一个对象变量的值另外一个就会一起改变,这就是我们常说的浅拷贝。而在深拷贝中,会开辟一个新的内存地址用来存放新对象的值,两个对象对应两个不同的内存地址 ,修改一个对象并不会对另外一个对象产生影响。接下来就让我们更细致的探究js中的深浅拷贝。

浅拷贝

实现浅拷贝的方法有多种,让我们先来看看js中提供的几个自带方法实现浅拷贝的的例子:

  • Object.assign()方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。注意:Object.assign()拷贝的是属性值,假如源对象的属性值是一个指向对象的引用,它也只拷贝那个引用值,来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

如上面例子,当拷贝的源对象的属性值是一个对象时,拷贝的只是对象的引用值,因此当修改属性值的时候两个对象的属性值都会发生更新

  • Array.prototype.slice()方法提取并返回一个新的数组,如果源数组中的元素是个对象的引用,slice会拷贝这个对象的引用到新的数组,来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

如上例所示,但源数组中的元素是对象引用时,slice拷贝的是这个对象的引用,因此当修改其中一个的值时,两个数组中的值都会发生改变

  • Array.prototype.concat()用于合并多个数组,并返回一个新的数组,和slice方法类似,当源数组中的元素是个对象的引用,concat在合并时拷贝的就是这个对象的引用,来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

除了上述js中自带方法实现的浅拷贝外,我们自己如何简单实现一个浅拷贝呢?来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

实现一个浅拷贝,就是遍历源对象,然后在将对象的属性的属性值都放到一个新对象里就ok了,是不是很简单呢?

深拷贝

先来介绍一个做深拷贝最简单粗暴的方法JSON.stringify()和JSON.parse()的混合配对使用,相信大家对这两个方法都是非常熟悉的,来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

上述例子可以看出,使用JSON.stringify()和JSON.parse()确实可以实现深拷贝,在新对象中修改对象的引用时,并不会影响老对象里面的值,那么,这么个方法是否就没有缺陷了呢?在JSON.stringify()做序列时,undefined、任意的函数以及symbol值,在序列化过程中会被忽略,这会在对象复制的时候导致什么后果呢?来看一个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

从例子中可以看到,当源对象中有undefine、function、symbol时,在序列化操作的时候会被忽略,导致拷贝生成的对象中没有对应属性及属性值。那么怎么自己去实现一个深拷贝呢?比较常见的方法就是通过递归,来看个例子:

寥寥数笔带你洞悉JavaScript深拷贝和浅拷贝

 

通过对需要拷贝的对象的属性进行递归遍历,如果对象的属性不是基本类型时,就继续递归,知道遍历到对象属性为基本类型,然后将属性和属性值赋给新对象。

总结

以上对js深拷贝和浅拷贝做了简单的介绍,在深拷贝的实现上也只介绍了最简单的实现形式,并未考虑复杂情况以及相应优化,想要对深拷贝有更深入的了解,需要大家花时间去深入研究,或者可以关注我后续文章的动态。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值