对象的浅拷贝与深拷贝

  • 什么是对象的拷贝?

    将一个对象赋值给另外一个对象, 我们称之为对象的拷贝

  • 什么是深拷贝, 什么是浅拷贝?

    我们假设将A对象赋值给B对象

  • 浅拷贝是指, 修改B对象的属性和方法会影响到A对象的属性和方法, 我们称之为浅拷贝

    以下几种情况都属于浅拷贝:

    1、默认情况下对象之间的直接赋值都是浅拷贝

    1. let A = {
    2. name: 'zyx',
    3. age: 20
    4. }
    5. let B = A
    6. console.log(B) // {name: "zyx", age: 20}
    7. //修改B的 name 属性
    8. B.name = 'ls'
    9. //A 也收到影响
    10. console.log(A) // {name: "ls", age: 20}
    11. console.log(B) // {name: "ls", age: 20}

    ​ 赋值操作(包括对象作为参数、返回值),不会开辟新的内存空间,他只是赋值了对象的引用.也就是除了B这个名字之外,没有其他的内存开销,修改了A也就影响了B,修改了B,也就影响了A.

    如图所示:

2、如果对象的属性包含了引用数据类型(数组、对象),那么哪怕不是直接赋值操作,而是开辟了一层新的内存空间,也就是说只拷贝了A对象的一层,这仍然属于浅拷贝。

  1. let A = {
  2. name: 'ls',
  3. age: 20,
  4. hobbies: ['dance','basketball','read'],
  5. dogs:{
  6. name: '大黄',
  7. color: 'yellow'
  8. }
  9. }
  10. let B = {}
  11. //定义一个函数,把A对象的属性复制一份给B
  12. function extend(obj1,obj2){
  13. for(var key in obj1){
  14. obj2[key] = obj1[key]
  15. }
  16. }
  17. extend(A,B)
  18. //修改B对象中的引用类型数据 ,A对象也收到影响
  19. B.dogs.color = 'red'
  20. B.hobbies[0] = 'sing'
  21. console.log(B)
  22. console.log(A)

运行截图如下:修改B对象中的引用类型数据 ,A对象也收到影响,属于浅拷贝

3、ES6中新增的 Object.assign() 也是对象的浅拷贝

Object.assign方法用于对象的合并,将源对象(source)的所有可枚举属性,复制到目标对象(target)。 Object.assign方法的第一个参数是目标对象,后面的参数都是源对象。 注意,如果目标对象与源对象有同名属性,或多个源对象有同名属性,则后面的属性会覆盖前面的属性。

  1. const obj1 = {a: {b: 1}};
  2. const obj2 = Object.assign({}, obj1);
  3. obj1.a.b = 2;
  4. obj2.a.b // 2

上面代码中,源对象obj1a属性的值是一个对象,Object.assign拷贝得到的是这个对象的引用。这个对象的任何变化,都会反映到目标对象上面。

4、扩展运算符(...)

利用扩展运算符可以在构造字面量对象时,进行克隆或者属性拷贝 ,属于浅拷贝

  1. var obj = {a:1,b:{c:1}}
  2. var obj2 = {...obj};
  3. obj.a=2;
  4. console.log(obj); //{a:2,b:{c:1}}
  5. console.log(obj2); //{a:1,b:{c:1}}
  6. obj.b.c = 2;
  7. console.log(obj); //{a:2,b:{c:2}}
  8. console.log(obj2); //{a:1,b:{c:2}}

5、Array.prototype.slice()

slice() 方法返回一个新的数组对象,这一对象是一个由 begin和 end(不包括end)决定的原数组的浅拷贝。原始数组不会被改变。

  • 深拷贝是指, 修改B对象的属性和方法不会影响到A对象的属性和方法, 我们称之为深拷贝

    以下几种情况都属于深拷贝:

    1、默认情况下一个对象的属性如果是基本数据类型, 那么进行复制(拷贝),都是深拷贝

    如果A对象的属性都是基本数据类型(Number、String等),此时要想深拷贝一份A给B,该怎么做呢,在这种要拷贝的对象A只有基本类型的数据时,只需要在内存中开辟一块空间存储B就行了。

    1. let A = {
    2. name: 'zyx',
    3. age: 20
    4. }
    5. let B = {}
    6. //定义一个函数,把A对象的属性复制一份给B
    7. function extend(obj1,obj2){
    8. for(var key in obj1){
    9. obj2[key] = obj1[key]
    10. }
    11. }
    12. extend(A,B)
    13. console.log(B) // {name: "zyx", age: 20}
    14. B.name = 'ls'
    15. console.log(B) // {name: "ls", age: 20}
    16. console.log(A) // {name: "zyx", age: 20}

    这样就实现了深拷贝,如下图所示:

2、如果要拷贝的对象本身又包含了引用数据类型,即对象又包含数组或者对象,层层嵌套的情况下,想要实现对象的深拷贝,可以采用递归的方式进行深拷贝。

  1. let A = {
  2. name: 'ls',
  3. age: 20,
  4. hobbies: ['dance','basketball','read'],
  5. dogs:{
  6. name: '大黄',
  7. color: 'yellow'
  8. }
  9. }
  10. let B = {}
  11. //定义一个函数,把A对象的属性复制一份给B
  12. function extend(obj1,obj2){
  13. for(var key in obj1){
  14. var item = obj1[key]
  15. if(item instanceof Array){
  16. obj2[key] = []
  17. extend(item,obj2[key])
  18. }else if(item instanceof Object){
  19. obj2[key] = {}
  20. extend(item,obj2[key])
  21. }else{
  22. obj2[key] = item
  23. }
  24. }
  25. }
  26. extend(A,B)
  27. B.dogs.color = 'red'
  28. B.hobbies[0] = 'sing'
  29. console.log(B)
  30. console.log(A)

运行发现,修改B对象的引用数据类型,不会影响到A对象,完成深拷贝

我们可以对深拷贝的代码进行封装优化

  1. function deepClone(obj){
  2. let cloneObj = {}
  3. for(let key in obj){
  4. if(typeof obj[key] === 'object'){
  5. cloneObj[key] = deepClone(obj[key])
  6. }else{
  7. cloneObj[key] = obj[key]
  8. }
  9. }
  10. return cloneObj
  11. }

3、通过JSON.stringify实现深拷贝

JSON.stringify()是目前前端开发过程中最常用的深拷贝方式,原理是把一个对象序列化成为一个JSON字符串,将对象的内容转换成字符串的形式再保存在磁盘上,再用JSON.parse()反序列化将JSON字符串变成一个新的对象。

  1. var obj1 = {
  2. a:1,
  3. b:[1,2,3]
  4. }
  5. var str = JSON.stringify(obj1)
  6. var obj2 = JSON.parse(str)
  7. console.log(obj2); //{a:1,b:[1,2,3]}
  8. obj1.a=2
  9. obj1.b.push(4);
  10. console.log(obj1); //{a:2,b:[1,2,3,4]}
  11. console.log(obj2); //{a:1,b:[1,2,3]}

本文原创,欢迎到我的博客踩踩~ 地址:https://www.cnblogs.com/zyxnb/

总结JavaScript对象的深浅拷贝的更多相关文章

  1. Javascript 对象复制(深浅拷贝)

    一.数据类型分类: 基本变量 引用类型 二.什么叫做指针指向 栈内存.堆内存.指针指向(如下红圈圈的斜线). 三.赋值.拷贝.引用区别? 赋值指一个变量赋予某个值,包含两种方式,一种是直接量,另一种, ...

  2. 关于Java的Object.clone()方法与对象的深浅拷贝

    文章同步更新在个人博客:关于Java的Object.clone()方法与对象的深浅拷贝 引言 在某些场景中,我们需要获取到一个对象的拷贝用于某些处理.这时候就可以用到Java中的Object.clon ...

  3. javascript简单实现深浅拷贝

    深浅拷贝知识在我们的日常开发中还算是用的比较多,但是之前的状态一直都是只曾听闻,未曾使用(其实用了只是自己没有意识到),所以今天来跟大家聊一聊js的深浅拷贝: 首先我们来了解一下javascript的 ...

  4. Javascript 中的深浅拷贝

    工作中经常会遇到需要复制 JS 数据的时候,遇到 bug 时实在令人头疼:面试中也经常会被问到如何实现一个数据的深浅拷贝,但是你对其中的原理清晰吗?一起来看一下吧! 为什么会有深浅拷贝 想要更加透彻的 ...

  5. JavaScript中的深浅拷贝

    深浅拷贝 在JS中,数据类型分为两类: ​ 简单数据类型:Number.Boolean.String.undefined ​ 引用数据类型:Array.Object.Function 简单数据类型通常 ...

  6. JavaScript对象的深浅复制

    前言 从层次上来看,对象的复制可以简单地分为浅复制和深复制,顾名思义,浅复制是指只复制一层对象的属性,不会复制对象中的对象的属性,对象的深复制会复制对象中层层嵌套的对象的属性. 在复制对象时,除了要复 ...

  7. Object.clone()方法与对象的深浅拷贝

    转载:[https://www.cnblogs.com/nickhan/p/8569329.html] 引言 在某些场景中,我们需要获取到一个对象的拷贝用于某些处理.这时候就可以用到Java中的Obj ...

  8. js对象的深浅拷贝

    JS数据类型可以分为(ES5,暂时不考虑ES6): 简单数据类型:Number.String.undefined.boolean 复杂数据类型:Object.Array 简单的数据类型,往往是赋值操作 ...

  9. 在vue项目中遇到关于对象的深浅拷贝问题

    一.问题 项目里新添加了一个多选的功能,其显示的数据都是从后端返回过来的,我们需要在返回来的数据外再额外添加一个是否选中的标记,我的选择是在返回正确的数据时将标记添加进去,然后push到数组中.然后就 ...

随机推荐

  1. asp.net core 3.x 模块化开发之HostingStartup

    我们希望将一个项目(dll)看做一个模块/插件,一个模块往往需要在应用启动时做一些初始化工作,比如向IOC容器添加一些服务,为应用配置对象添加自己的数据源:也希望在应用关闭时做一些收尾工作,asp.n ...

  2. 【记】创建 VirtualBoxClient COM 对象失败. 应用程序将被中断

    1. 在本地64位win7系统安装VirtualBox完,启动时提示错误 原因:兼容性造成的 按照下图显示修改VirtualBox快捷方式的兼容性 2. 启动虚拟机时,提示 点击弹出框的确定按钮后,接 ...

  3. 【移动测试】你的测试用例中,是否包含App前后台切换

    App前后台切换是我们平时常用的一个操作,比如:按手机的home键将应用置于后台.直接按手机电源键关闭屏幕或者通过最近打开的应用列表切换应用等,由此,我们可以得出结论:当app置于前台时,它的页面对我 ...

  4. zm吃包子

    [题目背景]: zm 喜欢上了吃包子. [题面描述]: zm 每天都要去买包子,但是为了减肥,zm 设置了一系列规则来控制他每天买包子的数量. 他随机了 n 个特殊字符串,然后用 n 个字符串来衡量接 ...

  5. SEATA 分布式事务入门DEMO

    Simple Extensible Autonomous Transacation Architecture,seata是简单的.可扩展.自主性高的分布式架构 SEATA Server Configu ...

  6. js 极简获取表单 元素 !

    let s =[]; $.each($('#formSearch input'),(m,n)=>{s.push(n)}); //示例获取表单所有 input 下滑线分割的 name 集合.set ...

  7. python类型检查和类型转换

    类型检查 type()用来检查值的类型 (整型.浮点型.布尔值.字符串.空值) 该函数会将检查的结果作为返回值返回,可以通过变量来接收函数的返回值 print(type(1)) # <class ...

  8. 头条面试竟然问我maven

    maven package和maven install 有什么区别? 你常用的maven命令有哪些? <dependencyManagement> 是干什么的? 还有用过其它构建工具吗? ...

  9. Core 定时任务之HangFire

    ASP.NET Core 使用 Hangfire 很简单,首先,Nuget 安装程序包 > install-package Hangfire -pre 然后ConfigureServices添加 ...

  10. 深入理解es6中的Promise

    https://www.jianshu.com/p/9e4af5b77253 https://zhuanlan.zhihu.com/p/30797777 https://segmentfault.co ...