当前位置: 首页 > news >正文

前端高频面试题:深拷贝和浅拷贝的区别?

前言

对于前端同学来说,对象的深拷贝和浅拷贝可以说是面试中最火热的题目之一了,今天我们一起来把它盘明白。

1、深拷贝和浅拷贝介绍

深拷贝和浅拷贝都是对对象进行拷贝,其主要区别是,在对象拷贝时,对引用数据类型的处理方式。

1.1 浅拷贝(Shallow Copy)

浅拷贝是指只复制对象本身和对象中的基本数据类型,对于引用类型的属性,只复制内存地址引用,不复制引用的对象本身

浅拷贝的特点:

  • 只拷贝第一层。
  • 对于基本数据类型,拷贝值。
  • 对于引用数据类型,拷贝引用,即新对象和原对象共享引用类型的属性。

1.2 深拷贝(Deep Copy)

深拷贝是指创建一个对象,这个对象的内容和原始对象完全相同,但它们是存储在不同的内存地址上的,这意味着,我们修改新对象,原始对象不受影响

深拷贝的特点:

  • 拷贝所有层级,对多层的属性进行循环递归拷贝。
  • 对于基本数据类型,拷贝值。(和浅拷贝相同)
  • 对于引用数据类型,创建一个新的引用对象,并循环拷贝到新对象中,也就是新对象和原对象分别引用独立的引用数据,不共享引用。

1.3 示例说明

举个例子:

constobj={a:1,b:[1,2,3],c:{d:4,e:5,}}constnewObj1=shallowClone(obj);console.log('newObj1:',newObj1);constnewObj2=deepClone(obj);console.log('newObj2:',newObj2);/** 打印结果: newObj1: { a: 1, b: [ 1, 2, 3 ], c: { d: 4, e: 5 } } newObj2: { a: 1, b: [ 1, 2, 3 ], c: { d: 4, e: 5 } } */

我这里并没有给出shallowClonedeepClone的实现代码,后文会分别详细介绍。

从打印结果来看,newObj1newObj2都是相同的,而不同的是,对于浅拷贝来说:

  • obj.b === newObj1.b,结果为true
  • obj.c === newObj1.c,结果为true

而对于深拷贝来说,它们前后都是不相等的。

  • obj.b === newObj.b,结果为false
  • obj.c === newObj.c,结果为false

2、浅拷贝的实现方式

2.1 展开运算符

可以用ECMAScript 2015新增特性,也就是ES6提供的语法三个点(...)展开运算符,它可以展开可迭代对象(数组、字符串、Map、Set 等),从而来实现浅拷贝,具体用法如下:

constnewObj={...obj};

2.2 Object.assign

Object.assignObject类自带的一个静态方法,可以将一个或多个对象中的可枚举(此属性的enumerable为true)自有属性(对象自身的,从原型上继承的不算)合并到目标对象中。

constnewObj=Object.assign({},obj1, obj2,obj3,...objN);

2.3 for…in + Object.prototype.hasOwnProperty

直接用for..in循环,配合Object.prototype.hasOwnProperty判断是否是自身的属性来进行拷贝。

functionshallowClone(obj){constnewObj={};for(letkeyinobj){if(obj.hasOwnProperty(key)){newObj[key]=obj[key];}}returnnewObj;}constobj={a:1,b:[1,2,3],c:{d:4,e:5,}};constnewObj=shallowClone(obj);console.log(newObj)

或者用Object.keys先拿到自身的属性的数组,然后forEach循环拷贝也可,方式有很多种,大家可以自行扩展。

2.4 扩展:展开运算符和Object.assign在实现浅拷贝上有什么区别?

展开运算符和Object.assign虽然可以都实现浅拷贝,但仍有细微的区别,比如在遇到gettersetter时,两者的表现不一样。来举个例子:

首先是展开运算符:

constobj1={geta(){console.log('getter')return1;},seta(val){console.log('setter')}}constobj2={a:2,}constnewObj1={...obj1,...obj2};console.log(newObj1);console.log(newObj1.a)console.log('------------------------');constnewObj2={...obj2,...obj1};console.log(newObj2);console.log(newObj2.a)/** * 打印结果: getter { a: 2 } 2 ------------------------ getter { a: 1 } 1 */

从打印结果可以看出,展开运算符在拷贝时有如下特点:

  1. 合并时不会执行setter
  2. 合并后取值时会按照合并的先后顺序,后合并的值优先级更高

然后是Object.assign

constobj1={geta(){console.log('getter')return1;},seta(val){console.log('setter')}}constobj2={a:2,}constnewObj=Object.assign(obj1,obj2);console.log(newObj);console.log(newObj.a)console.log('------------------------');constnewObj1=Object.assign(obj2,obj1);console.log(newObj1);console.log(newObj1.a);/** * 打印结果: setter { a: [Getter/Setter] } getter 1 ------------------------ getter { a: 1 } 1 */

从打印结果可以看出,``Object.assign`在拷贝时有如下特点:

  1. 同名属性和同名gettersetter合并时,会执行setter,而同名gettersetter同名属性合并时却不会执行setter
  2. 无论同名属性和同名gettersetter的合并先后顺序如何,最终访问只会访问到getter里面的值,只是从控制台里看的效果不一样而已。

3、深拷贝的实现方式

3.1 JSON.parse(JSON.stringify(obj))

  • JSON.stringify:将一个对象序列化成一个JSON字符串,包括嵌套的对象属性。
  • JSON.parse:将一个JSON反序列化成为一个 JS 对象。

由于在内存中JSON 字符串的地址都是独立的,和原始对象不是同一个地址,所以我们就能通过JSON.parse解析出一个新对象了。

下面看一下用这种方式实现深拷贝的优缺点:

优点:

  1. 简单易用:语法JSON.parse(JSON.stringify(obj)),用起来非常简单。
  2. 跨平台:在不同平台和环境都能用,甚至是其它语言也有提供对应的实现,比如Java
  3. 兼容性好:各大浏览器都支持。

缺点:

  1. 无法处理特殊对象类型,比如函数、正则表达式、日期对象等
    • 拷贝的时候会丢失函数和 undefined。
    • 时间对象 Date 会变成字符串形式。
    • RegExp、Error 对象会变成空对象。
    • NaN、Infinity、-Infinity会变成 null。
    • 等等…。
  2. 无法处理循环引用,比如在一个对象中,a引用了b,b引用了c,而c又引用了a,出现这种情况调用JSON.parse(JSON.stringify(obj))会报错。

3.2 借助第三方库

实现深拷贝,一般我们会借助第三方库实现,比如lodash,lodash提供了一个cloneDeep的方法实现深拷贝。

const_=require('lodash');constobj={a:[{b:2}]};constres=_.cloneDeep(obj);console.log(res);// 输出:{ a: [ { b: 2 } ] }

3.3 手撸一个深拷贝方法

深拷贝其实实现起来要写完整,还是挺复杂的,要处理函数数组正则,甚至是symbolbuffer等,但对于面试来说,我们写个简单版本就行啦。

手动实现深拷贝有两个关键点:

  • 对象是以keyvalue键值对的方式存储的,所以要拷贝它们必须要用循环
  • 既然要深拷贝,相较于只拷贝最外层浅拷贝,就需要用递归或循环拷贝N层

废话不多说,直接上完整代码。

constisObj=(target)=>typeoftarget==='object'&&target!==nullfunctiondeepClone(obj,hash=newWeakMap()){if(!isObj(obj))returnobjif(hash.has(obj))returnhas.get(obj)consttarget=newobj.constructor()hash.set(obj,target)Object.keys(obj).forEach((key)=>{target[key]=deepClone(obj[key],hash)})returntarget}constobj={a:[{b:2}]}constres=deepClone(obj)console.log(res)// 输出:{ a: [ { b: 2 } ] }

我们用一个WeakMap来处理循环引用,然后通过拿到对象引用的constructor来复制对象,这样我们就省去了判断不同对象类型这一步,会简单很多,然后forEach循环递归复制就好啦。

虽然可能有人会说用constructor比较粗糙,但这是比较简洁的写法,我们面试的时候大可不必这么较真,只要知道它的核心思路和原理就行啦!

4、如何选择浅拷贝和深拷贝?

浅拷贝能共享数据,节约内存,性能高

深拷贝实现数据隔离,数据更安全。但要注意循环引用问题特殊对象的处理以及对象层级过深带来的性能问题

在平常开发中,数据拷贝几乎设计不到性能问题,所以如果不介意引用数据共享,选浅拷贝,需要引用数据相互独立,选深拷贝。

小结

  1. 先介绍了深拷贝深拷贝和浅拷贝的区别。它们的主要区别在于拷贝时对引用数据类型的处理,浅拷贝是共享引用,而深拷贝是复制出一个新对象,和原对象相互独立,没任何关系。

  2. 然后介绍了浅拷贝和深拷贝的主要实现方式。浅拷贝的实现方式主要有展开运算符Object.assignfor..in + Object.prototype.hasOwnProperty等,还扩展介绍了展开运算符和Object.assign的区别,主要是体现在复制时对属性访问器gettersetter的处理方式不同,而深拷贝的实现方式主要有JSON.parse(JSON.stringify(obj))借助第三方库,比如 lodash手撸一个深拷贝方法(注意对循环引用的处理)

  3. 最后介绍了在实际开发中如何选择浅拷贝和深拷贝。主要选择方式是,不介意引用数据共享,选浅拷贝,需要引用数据相互独立,选深拷贝。

http://www.cnnetsun.cn/news/140433.html

相关文章:

  • python_django农业生产环境下的土壤与气候监控数据处理系统 可视化999i002k
  • python_django基于web平台的社区医疗服务平台_nk5a3uy7
  • 容错不是选修课,而是生死线:工业控制Agent必须具备的4项容错能力
  • python_django基于大数据技术旅游景点数据分析推荐系统现_wrqk1aes
  • 基于安卓的健身记录APP
  • 测试方法创新:从自动化到智能化的转型之路‌
  • 基于多智能体深度强化学习的车联网通信资源分配优化探索
  • 小程序毕设项目推荐-基于微信小程序的学生宿舍管理系统基于springboot+微信小程序的高校学生公寓道闸管理平台的设计与实现【附源码+文档,调试定制服务】
  • 小程序毕设项目推荐-基于springboot+vue的微信小程序的快递代取系统的设计与实现基于springboot+微信小程序的快递代取系统的设计与实小程序【附源码+文档,调试定制服务】
  • 小程序毕设项目推荐-基于微信小程序的宠物服务系统基于springboot+微信小程序的宠物服务系统小程序【附源码+文档,调试定制服务】
  • 给AI装上“海马体”:三层类人记忆架构如何让多Agent系统真正懂你
  • 影刀RPA黑科技:自动分析Zozone用户消费行为,精准营销一键搞定![特殊字符]
  • 【教育 AI 突破性进展】:3个真实案例揭示学情分析如何提升教学效率40%+
  • 通达信成交额优化公式指标
  • 数字电路高阶部分<1>数字电路里的“找不同”:隐含表到底藏着什么玄机?
  • Java开发必备:Maven集成IDEA详细教程
  • 喜报!凯云成为北京软件和信息服务业协会第十一届理事会会员单位,并荣获“双软认证”
  • 昨晚被消消乐广告气到,回家我直接用 AI 复刻了“究极进化版”俄罗斯方块!
  • 首创ACE具身研发范式,大晓机器人构建具身智能开放新生态
  • 41、Linux多核处理器性能优化与调试指南(上)
  • 44、深入探索GDB调试:数据检查、函数调用与模板调试技巧
  • WebDriver+Selenium实现浏览器自动化
  • QUIC协议:下一代互联网传输协议的技术革新与应用前景
  • 基于单片机的智能灯光控制系统设计
  • 贪心算法专题(三):负重前行,不如从头再来——「最大子序和」
  • STL容器——String容器
  • Mal-PEG4-NHS ester,化学特性及其在蛋白质修饰与生物分子功能化研究中的应用
  • 详细分析一下 国富论里里面 十一章 论 地租
  • 现在 夸脱小麦 多少 盎司白银
  • Java Web html 图书管理系统系统源码-SpringBoot2+Vue3+MyBatis-Plus+MySQL8.0【含文档】