什么是浅拷贝和深拷贝,如何用 js 代码实现?

〇、简介和对比

  • 简介

浅拷贝:只复制原始对象的第一层属性值。

  如果属性值是值类型,将直接复制值,本值和副本变更互不影响;

  如果是引用数据类型,则复制内存地址,因此原始对象和新对象的属性指向相同的内存地址,改变任一值,另一变量值也会同步变更。

深拷贝:递归地复制原始对象的所有层级。

  每一个属性值都会在新的对象中重新创建,无论变量是值类型还是引用类型,修改新对象不会影响原对象

  • 实现方法

浅拷贝:可以通过 Object.assign()、扩展运算符(...)、Array.prototype.slice()、Array.prototype.concat() 等方法来实现浅拷贝。

深拷贝:可以通过 JSON.stringify() 与 JSON.parse() 的组合、递归函数、或使用一些库如 jQuery.extend() 方法来实现深拷贝。

  • 适用场景

浅拷贝:当对象属性不包含引用类型或不需要深层结构复制时使用。或者说,对象结构简单,或者您希望拷贝后的对象与原对象保持一定的关联性,可以选择使用浅拷贝。

深拷贝:当对象有多层嵌套或需要完全独立的拷贝时使用。另外,当对象结构较为复杂,包含多层嵌套的引用类型时,考虑使用深拷贝以确保数据的独立性。

  • 注意事项

浅拷贝:

  拷贝后的对象与原对象引用类型的属性和值的共享问题;

  性能开销较深拷贝小,仅复制一层属性;

  没有处理循环引用的机制;

  无法复制原始对象的深层属性。

深拷贝:

  需要考虑性能消耗以及特殊类型(如不能复制 Function、Error 等)的处理,且可能受浏览器支持限制;

  性能开销较大,特别是对于大型对象,递归复制所有层级;

  需要特殊处理以避免无限递归,如使用 WeakMap 来跟踪复制过程中已经复制过的引用。

 一、值类型变量无需区分浅拷贝和深拷贝

值类型数据的值存放在栈中,而引用类型的地址存放栈中,数据存放在堆中。变量浅拷贝实际就是复制的栈中的数据,对于值类型来说,浅拷贝就是直接拷贝值,等效于深拷贝。因此值变量不区分浅拷贝和深拷贝。关于值类型和引用类型

如下示例代码,针对值类型的 int 进行浅拷贝,修改副本的值,也不影响原值:

// 值类型 int
int i1=10;
int i2=i1;
i2=5; // 重新给 i2 赋值
console.log(i1,i2); // 10 5

二、引用类型的浅拷贝

对于引用类型来说,它仅仅把地址保存在栈中,实际的值则在堆中。关于值类型和引用类型

浅拷贝就是针对栈中的地址的拷贝,当修改变量的值时,不影响实际的地址,当堆中的值有多个引用时,其他地址对应的值也会随之变更。

下边是几个浅拷贝的方法。

2.1 直接通过等号 = 赋值,复制的是原值的地址

先看一个简单的引用类型的示例代码,修改副本的值,也会影响原值:

// 引用类型的 object 对象
var a = { name: 'Marry' };
var b = a;          // 将栈中的地址赋值给新的变量 b
b.name = 'Jone';    // 通过副本的地址修改堆中的值,会导致其他引用此地址的表量值一起变更
console.log(a.name) // Jone
console.log(b.name) // Jone
// 其实变量 a 和 b 栈中的地址一样,都指向同一个堆中的值

再来看下另外一个关于对象数组的例子:

const sourceArray = [{"name":"Zhangsan"}, {"name":"Lisi"}];
const targetArray1 = sourceArray;
targetArray1.push({"name":"Wangwu"});
targetArray1[0].name="Zhangsan---";
console.log(sourceArray);
console.log(targetArray1);

通过 = 赋值的变量,就是将原值的地址赋值给了副本,两个变量其实是指向同一个数组的地址,因此修改任一变量的值,另外一个也会随之变动。

如下输出结果,对新的数组变量操作,原数组也随之变动

  

2.2 另外四种种浅拷贝的方法:slice()、concat()、[...ArrayName]、Object.assign([],ArrayName)

这四种浅拷贝的效果是相同的,都是复制了当前数组的全部引用地址。若是新增的值,对原数组无影响;若是对原来已有的值进行修改,则原数组的对应的值也会随之变动。

特别注意:当要拷贝的对象为值类型的数组,这四种方法拷贝的直接就是数组中各项的值,就是深拷贝的效果。

如下代码四种方式,操作副本的值,添加新值和修改原副本的值,对原值会有不同效果:

const sourceArray = [{"name":"Zhangsan"}, {"name":"Lisi"}];
const targetArray1 = sourceArray.slice();             // 第一种
//const targetArray1 = sourceArray.concat();          // 第二种
//const targetArray1 =[...sourceArray];               // 第三种
//const targetArray1 = Object.assign([],sourceArray); // 第四种
targetArray1.push({"name":"Wangwu"}); // 往对象数组中添加一个对象
targetArray1[0].name="Zhangsan---";   // 修改浅拷贝副本中的第一个对象值
console.log(sourceArray);
console.log(targetArray1);

输出结果:

  

若要避免多副本修改互相的影响,就需要深拷贝,下面就来看下深拷贝的实现方式。

三、引用类型的深拷贝

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

还参考上一章节的例子,将 sourceArray 进行深拷贝:

const sourceArray = [{"name":"Zhangsan"}, {"name":"Lisi"}];
const targetArray1 = JSON.parse(JSON.stringify(sourceArray)); // 深拷贝
targetArray1.push({"name":"Wangwu"}); // 编辑副本数组
targetArray1[0].name="Zhangsan---";
console.log(sourceArray);
console.log(targetArray1);

查看结果可知,原数组的值并未发生变更:

  

3.2 通过递归函数实现

如下代码中的递归函数 deepClone():

window.onload = function () {
    const sourceArray = [{"name":"Zhangsan"}, {"name":"Lisi"}];
    const targetArray1 = deepClone(sourceArray);
    targetArray1.push({"name":"Wangwu"});
    targetArray1[0].name="Zhangsan---";
    console.log(sourceArray);
    console.log(targetArray1);
}
function deepClone(source) {
    if (typeof source !== 'object' || source == null) { // 当入参不是对象或者为空时直接返回
        return source;
    }
    const target = Array.isArray(source) ? [] : {}; // 判断输入变量为对象数组还是对象
    for (const key in source) {
        // Object.prototype.hasOwnProperty.call(source, key) 是一个 JavaScript 方法
        // 用于检查对象(source)是否具有指定的属性(key)
        // 如果对象具有该属性,则返回 true,否则返回 false
        if (Object.prototype.hasOwnProperty.call(source, key)) {
            if (typeof source[key] === 'object' && source[key] !== null) {
                target[key] = deepClone(source[key]); // 若属性值仍为对象,则进行递归操作
            } else {
                target[key] = source[key];
            }
        }
    }
    return target;
}

兼容多种数据类型的递归方法:

function deepClone(source, cache){
    if (!cache) {
        cache = new Map()
    }
    if (source instanceof Object) { // 不考虑跨 iframe
        if (cache.get(source)) { return cache.get(source) }
        let result
        if (source instanceof Function) {
            if (source.prototype) { // 有 prototype 就是普通函数
                result = function () { return source.apply(this, arguments) }
            } else {
                result = (...args) => { return source.call(undefined, ...args) }
            }
        } else if (source instanceof Array) {
            result = []
        } else if (source instanceof Date) {
            result = new Date(source - 0)
        } else if (source instanceof RegExp) {
            result = new RegExp(source.source, source.flags)
        } else {
            result = {}
        }
        cache.set(source, result)
        for (let key in source) {
            if (source.hasOwnProperty(key)) {
                result[key] = deepClone(source[key], cache)
            }
        }
        return result
    } else {
        return source
    }
}

3.3 使用 jQuery.extend()

可通过参数控制是否为深拷贝,语法:

$.extend(deepCopy, target, object1, [objectN])
// deepCopy 为 true,表示深拷贝
//   结果为对象数组:[{"name":"Zhangsan"}, {"name":"Lisi"}]
// deepCopy 为 false,表示浅拷贝
//   结果为对象:{{"name":"Zhangsan"}, {"name":"Lisi"}}
//   不能直接进行 push 操作
// target 目标对象,即将后续一个或多个对象的值,全部合并至此对象

const sourceArray = [{"name":"Zhangsan"}, {"name":"Lisi"}];
const targetArray1 = $.extend(true, [], sourceArray); // 深拷贝
targetArray1.push({"name":"Wangwu"});
targetArray1[0].name="Zhangsan---";
console.log(sourceArray);
console.log(targetArray1);

结果为:

  

文章转载自:橙子家

原文链接:https://www.cnblogs.com/hnzhengfy/p/18152689/CS_Shallow_DeepCopy

体验地址:引迈 - JNPF快速开发平台_低代码开发平台_零代码开发平台_流程设计器_表单引擎_工作流引擎_软件架构

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/719805.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

小规模自建 Elasticsearch 的部署及优化

本文将详细介绍如何在 CentOS 7 操作系统上部署并优化 Elasticsearch 5.3.0,以承载千万级后端服务的数据采集。要使用Elasticsearch至少需要三台独立的服务器,本文所用服务器配置为4核8G的ECS云服务器,其中一台作为 master + data 节点、一台作为 client + data 节点、最后一…

【学习Docker】

学习Docker可以分为几个步骤和阶段,以下是一个建议的学习路径,适合初学者到进阶用户: ### 1. 理解基本概念 - **容器化与虚拟化**:了解容器化与传统虚拟化之间的区别,容器的轻量级和效率。 - **Docker组件**&#xff…

基于EXCEL数据表格创建省份专题地图

1 数据源 随着西藏于5月1日发布2022年一季度经济运行情况,31省份一季度GDP数据已全部出炉。 总量方面,粤苏鲁稳居前三;增速方面,23省份高于“全国线”,新疆表现最佳,增速达到7.0%。 表格表现数据不够直观…

使用 Dapper 创建 Blazor Server SPA

介绍 Blazor 是 Microsoft 构建的一个新框架,用于使用 .NET 代码库创建交互式客户端 Web UI。 Dapper 是一个微型 ORM(对象关系映射器),可帮助将本机查询输出映射到领域类。它是由 StackOverflow 团队构建并作为开源发布的高性能…

Ubuntu 使用Vscode的一些技巧 ROS

Ubuntu VSCode的一些设置(ROS) 导入工作空间 推荐只导入工作空间下的src目录 如果将整个工作空间导入VSCode,那么这个src就变成了次级目录,容易在写程序的时候把本应该添加到具体工程src目录里的代码文件给误添加到这个catkin_w…

如何使用Gitmails在版本控制主机中收集Git提交邮件

关于Gitmails Gitmails是一款能够在Git版本控制主机服务中收集Git提交电子邮件的信息收集工具,该工具可以帮助广大研究人员扫描和识别Git提交中包含的作者名称、电子邮件配置和版本控制主机服务是否存储了多个项目。 想要了解网络安全,学习网络安全知识…

SpringMVC系列九: 数据格式化与验证及国际化

SpringMVC 数据格式化基本介绍基本数据类型和字符串自动转换应用实例-页面演示方式Postman完成测试 特殊数据类型和字符串自动转换应用实例-页面演示方式Postman完成测试 验证及国际化概述应用实例代码实现注意事项和使用细节 注解的结合使用先看一个问题解决问题 数据类型转换…

数据中心分类和类别综合指南

数据中心可根据其规模、功能、所有权、层级和部署方法进行分类。以下是一些典型的数据中心类别和分类。 数据中心的分类和分级 根据尺寸 1. 小型数据中心:通常是为了满足对IT基础设施需求较少的组织或小型企业的需求而创建的。与大型数据中心相比,小型…

【Java学习笔记】枚举类与泛型

枚举类型是一种特殊的数据类型,之所以特殊,是因为它既是一种类(Class)类型,却又比类类型多了一些特殊的约束,但正是因为这些约束的存在,也造就了枚举类型的简洁性、安全性、便捷性。 泛型,即“参数化类型”…

【SPIE独立出版 | 往届均已完成EI检索】2024云计算、性能计算与深度学习国际学术会议(CCPCDL 2024)

2024云计算、性能计算与深度学习国际学术会议(CCPCDL 2024) 2024 International conference on Cloud Computing, Performance Computing and Deep Learning *CCPCDL往届均已完成EI检索,最快会后4个半月完成! 一、重要信息 大会官网:www…

python-开学?

[题目描述] 小执:终于可以开学啦!好开心啊! 小理:你没看新闻吗,开学日期又延后了。 小执:𝑁𝑂𝑂𝑂𝑂𝑂𝑂𝑂&am…

Vue01-前端概述

一、前端核心分析 1.1、概述 Soc原则:关注点分离原则 Vue 的核心库只关注视图层,方便与第三方库或既有项目整合。 HTML CSS JS : 视图 : 给用户看,刷新后台给的数据 网络通信 : axios 页面跳转 : v…

Java聚合快递系统对接云洋系统快递小程序APP公众号系统源码

快递小程序的深度解析与未来展望 🚚 引言:快递行业的变革与挑战 在数字化浪潮的推动下,快递行业正经历着前所未有的变革。随着电商的蓬勃发展,快递业务量呈爆发式增长,而传统的快递管理方式已难以满足日益增长的需求。…

我用chatgpt写了一款程序

众所周知,Chatgpt能够帮助人们写代码,前几天苏音试着完全用Chatgpt写一款Python程序 有一句话我很赞同,未来能代替人的不是AI,是会使用AI的人。 最终,写下来效果还不错,完全提升了我的办公效率。 开发前…

计算机跨考现状,两极分化现象很严重

其实我觉得跨考计算机对于一些本科学过高数的同学来说有天然的优势 只要高数能学会,那计算机那几本专业课,也能很轻松的拿下,而对于本科是文科类的专业,如果想跨考计算机,难度就不是一般的大了。 现在跨考计算机呈现…

了解Java的LinkedBlockingQueue

了解Java的LinkedBlockingQueue LinkedBlockingQueue是一个基于链接节点的有界阻塞队列。它实现了BlockingQueue接口,可以在多线程环境中安全地进行插入、移除和检查操作。LinkedBlockingQueue的容量可以在创建时指定,如果未指定,则默认容量…

AI绘画stable diffusion 模型介绍及下载、使用方法,超全的新手入门教程建议收藏!

大家好,我是画画的小强 今天我将继续分享AI绘画Stable Diffusion的模型、参数含义等,分享给各位朋友一起学习。 一、模型 Stable difusion 模型就是所谓的大模型,用来控制整个画面的风格走势的。 打开webui页面,可以看到大模型…

安卓实现圆形按钮轮廓以及解决无法更改按钮颜色的问题

1.实现按钮轮廓 在drawable文件新建xml文件 <shape xmlns:android"http://schemas.android.com/apk/res/android"<!--实现圆形-->android:shape"oval"><!--指定内部的填充色--><solid android:color"#FFFFFF"/><!-…

Mongodb介绍及window环境安装

本文主要内容为nosql数据库-MongoDB介绍及window环境安装。 目录 什么是MongoDB&#xff1f; 主要特点 MongoDB 与Mysql对应 安装MongoDB 下载MongoDB 自定义安装 创建目录 配置环境变量 配置MongoDB服务 服务改为手动 启动与关闭 安装MongoDB Shell 下载安装包 …

高考分数限制下,选好专业还是选好学校?

高考分数限制下&#xff0c;选好专业还是选好学校&#xff1f; 高考作为每年一度的盛大考试&#xff0c;不仅关乎学生们的未来&#xff0c;更承载了家庭的期望。2004年高考刚刚结束&#xff0c;许多考生和家长已经开始为填报志愿而焦虑。选好学校和专业&#xff0c;直接关系到…