首页 > 其他分享 >什么是浅拷贝和深拷贝,如何用 js 代码实现?

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

时间:2024-06-17 16:43:48浏览次数:28  
标签:name 对象 代码 js source sourceArray 拷贝 targetArray1

〇、简介和对比

  • 简介

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

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

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

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

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

  • 实现方法

浅拷贝:可以通过 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://segmentfault.com/a/1190000041847063     https://www.cnblogs.com/tangjiao/p/9313829.html                  

标签:name,对象,代码,js,source,sourceArray,拷贝,targetArray1
From: https://www.cnblogs.com/hnzhengfy/p/18152689/CS_Shallow_DeepCopy

相关文章

  • dlib库实现摄像头疲劳检测(附源代码)
     目录1.导入库2.定义添加中文文本的函数3.定义绘制眼框凸包的函数4.定义检测闭眼的函数5.定义检测大笑的函数6.构造检测器7.计算纵横比8.疲劳判定9.可视化输出 本代码将实现通过分析人脸的关键点进行疲劳检测,若持续闭眼则提示危险;考虑到大笑时也会眯眼,在同时检......
  • 解释一下这段代码 npm i --no-save --legacy-peer-deps react@17
    这段命令是用于在Node.js项目中安装React17版本的一个指定操作,具体各部分含义如下:npmi或npminstall:这是用于在Node.js项目中安装包的命令,会根据package.json文件中的dependencies或devDependencies安装所有依赖,如果没有指定特定包,则会安装所有列出的依赖。--no-save:这......
  • 经典Prompt欣赏 - 使用伪代码Prompt来让GPT生成绘本小蝌蚪找妈妈
    今天无意中发现尹相志老师用GPT-4o伪代码生成绘本的演示(https://www.youtube.com/watch?v=3rb-54Q5fig),结果让我大开眼界。这种全新的方法,不仅极大简化了复杂的创作过程,让人惊叹不已。让我们先来看看部分生成效果图:Prompt为了方便阅读,我将这个Prompt翻译成了......
  • 以前功能正常的代码突然出现布局问题
    我的网站使用相同的CSS文件已超过十年,从未出现过任何问题,但大约24小时前,我的网站布局出现了乱码,尽管我只是为我试图用于大写字母的新图片添加了一个单独的类,迄今为止我只在一个页面上进行了测试,但整个网站(每个页面)都出现了乱码,即使我恢复了原始CSS文件且未做任何新的修改,情......
  • 经典Prompt欣赏 - 使用伪代码Prompt来让GPT生成绘本小蝌蚪找妈妈
    今天看到尹相志老师用GPT-4o伪代码的方式做绘本生成,实际测试了下,被惊呆了。https://www.youtube.com/watch?v=3rb-54Q5fig先看生成的部分效果图:Prompt为了方便阅读,我把这个Prompt翻译成简体中文了,完整Prompt如下,实际执行时:碰到GPT-4o停下来时,输入“继续”即可......
  • three.js 从零学习
    基本概念场景用来呈现内容的容器我理解就是类似canvas相机记录场景中呈现的内容一般分类两大类1.正投影相机  所有内容同等大小呈现处理2.透视相机 符合人眼逻辑近大远小渲染器决定场......
  • [JS] 动态执行JS与修改词法作用域
    相关可行的操作eval:同步执行,当前作用域;setTimeout:异步执行,全局作用域;第1个参数可以传入函数对象,也可以传入字符串,即要执行的代码。script:同步执行,全局作用域;创建script标签,并设置innerHTML为要执行的代码。Function:同步执行,全局作用域。Function构造函......
  • Diffusers代码学习:LCM 图生图
    要将LCM用于图像到图像,需要将支持的LCM模型的Checkpoint加载到[UNet2DConditionModel]中,并用[LCMscheduler]替换scheduler程序。然后,可以像往常一样使用管道,并传递文本提示和初始图像,只需4个步骤即可生成图像。# 以下代码为程序运行进行设置importosos.environ["HF_ENDP......
  • js基础文档
    数据类型数据分为基本数据类型(String,Number,Boolean,Null,Undefined,Symbol)和对象数据类型。基本数据类型的特点:直接存储在栈(stack)中的数据引用数据类型的特点:存储的是该对象在栈中引用,真实的数据存放在堆内存里引用数据类型在栈中存储了指针,该指针指向堆中该实体的起......
  • VSCode使用svn代码管理工具,初次检出失败
    打开vscode,若要使用SVN需要下载相应的插件。 2.安装之后,需要对SVN插件进行配置,配置本地SVN的命令行执行文件地址。点击左下角齿轮,选择“设置Settings”。"svn.path":"C:/ProgramFiles/TortoiseSVN/bin/svn.exe"3.如图所示。设置完毕后重启VSCODE。4.按ctrl+shift......