前端的浅拷贝和深拷贝区别_解构赋值是深拷贝还是浅拷贝

前端的浅拷贝和深拷贝区别_解构赋值是深拷贝还是浅拷贝一、浅拷贝和深拷贝的区别数据都是存储在内存当中,而我们调用数据的时候都是通过地址来调用数据。对于浅拷贝来说,比如一个数组,只要我们修改了一个拷贝数组,那么原数组也会改变!vara=[0,1,2,3,4];varb=a;console.log(b);//[0,1,2,3,4]console.log(a);//[0,1,…

大家好,又见面了,我是你们的朋友全栈君。如果您正在找激活码,请点击查看最新教程,关注关注公众号 “全栈程序员社区” 获取激活教程,可能之前旧版本教程已经失效.最新Idea2022.1教程亲测有效,一键激活。

Jetbrains全系列IDE稳定放心使用

引言

什么是基本类型值和引用类型值?

基本类型值指的是简单的数据段,而引用类型值指那些可能由多个值构成的 对象

var name = "John"; // 基本类型值

var obj = new Object(); 
obj.name = "John"; 
// obj 为引用类型值

复制变量

在复制变量中,对于基本类型值来说,两者是互不影响的

var num = 1;
var num1 = num; // num1 = 1;

var num1 = 3; // num还是1,不会变

对于引用类型值来说,复制分为两种,一种是浅拷贝,一种是深拷贝。

一、浅拷贝和深拷贝的区别

数据都是存储在内存当中,而我们调用数据的时候都是通过地址(指针)来调用数据。

对于浅拷贝来说,比如一个数组(数组是一种对象),只要我们修改了一个拷贝数组,那么原数组也会改变!

	var a = [0,1,2,3,4];
	var b = a;
	console.log(b);   //[0,1,2,3,4]
	console.log(a);   //[0,1,2,3,4]
	b[0] = 2;
	console.log(b);   //[2,1,2,3,4]
	console.log(a)    //[2,1,2,3,4]

因为他们引用的是同一个地址的数据!拷贝的时候并没有给b数组创造独立的内存,只是把a数组指向数据的 指针 拷贝给了b!
而深拷贝就与其相反,将会给b数组创造独立的内存,并且将a数组的内容一一拷贝进来,两者互不影响。

二、浅拷贝和深拷贝的原理(参考了某大大的博客并引用了他的图,手动滑稽~~)

浅拷贝和深拷贝一般是对于引用类型值(如对象)来讲的,而基本类型值(如undefined、null、number、string、boolean以及es6新增的Symbol),只要是复制,就一定是另开辟以存储空间!

①基本数据类型存储:名值都存在栈内存中

如let a = 1;

这里写图片描述

当let b = a时,b复制了a,栈内存会新开辟一个内存

这里写图片描述

所以两者互不影响,修改谁都没问题!

②引用数据类型:名存在栈内存中,值存在于堆内存中,但是栈内存会提供一个引用的地址指向堆内存中的值

以上面浅拷贝的例子画图:
  这里写图片描述

b复制了a后,b是引用了a的值的存储地址,而不是把a的值复制了下来!

这里写图片描述

所以改了b,a也会受到影响。

三、实现深拷贝

方法1、层级拷贝,用递归实现。

何为层级?如,var a = [0,1,2,[3,4],5];
 a有两层,第一层是[3,4]外围的一层;第二层是[3,4];

如果我们深拷贝的时候,这样,
	function deepClone(obj){
		let objClone = Array.isArray(obj)?[]:{};
		for(let i in obj){
			objClone[i] = obj[i];
		}
		return objClone;
	}
	var a = [0,1,2,[3,4],5];
	var b = deepClone(a);

我们得到的b它的首层是另辟存放空间的,也就是上面所说的[3,4]的外面一层存放在另开辟的存储空间中。但是[3,4]这第二层他仍然是引用数据地址,因为上面的遍历只遍历了首层

所以我们应该在上面的代码中加入判断,当碰到数组、对象等时递归函数。

	function deepClone(obj){
		//定义对象来判断当前的参数是数组还是对象
		let objClone = Array.isArray(obj)?[]:{};
		//如果obj存在并且为对象		
		if(obj&&typeof obj == "object"){
			for(let key in obj){
				if(obj.hasOwnProperty(key)){
					//如果obj的子元素为对象,那么递归(层级遍历)
					if(obj[key]&&typeof obj[key] == "object"){
						objClone[key] = deepClone(obj[key]);
					}else{
					//如果不是,直接赋值
						objClone[key] = obj[key];
					}
				}
			}
		}	
		return objClone;
	}

方法2、JSON解析(底层原理也是层级遍历)

依旧是上面的例子,var a =  [0,1,2,[3,4],5];

var b = JSON.parse(JSON.stringify(a));

b就是拷贝的结果,修改b不影响a。但是这种方法也有缺陷:

  1. 无法复制函数
  2. 原型链没了,对象就是object,所属的类没了。
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/187294.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 计算机考研各省份学校,想考研究生,哪个省份的高校更容易考上?

    计算机考研各省份学校,想考研究生,哪个省份的高校更容易考上?广东省2020年报考人数为17.4万,比去年增长24.3%。考研人数在过去的5年里飞速增长,自2015年的5.16万到2020年的17.4万,五年时间里翻了两倍多,涨幅在国内名列前茅。山东省2020年硕士研究生招生考试准考人数共313190人,比去年增加58704人,增幅为23.1%,同样每年新增几万的考研人。江苏省2020年共有24.9万名考生报名参加硕士研究生考试,比去年增长17.7%,再创历…

    2022年5月22日
    74
  • micro f1和macro f1_correct score

    micro f1和macro f1_correct score首先,明确一个概念,精确率(Precision)和召回率(Recall)并不是只有在目标检测中才有的,是所有任务涉及到分类的都有。而且P和R是针对于每一类来说的,每一个类别都有自己的准确率和召回率,计算每一个类别时,该样本即为正样本,其他样本统一为负样本来计算。首先,先说F1score,它其实是用来评价二元分类器的度量。F1是针对二元分类的,那对于多元分类器,有没有类似F1score的度量方法呢?那就是这里提到的micro-F1和macro-F1macro-F1…

    2022年10月10日
    3
  • mongovue 导入mysql_MongoVUE简单操作手册「建议收藏」

    mongovue 导入mysql_MongoVUE简单操作手册「建议收藏」本文转自网络,对疑问的地方做了修改MongoVUE是个比较好用的MongoDB客户端,需要注册,但是可以变成永久使用,一、基础操作新增一个连接进入的界面形如二、进阶操作1、查看所有数据,会有三种格式的观看方式,分别为树形,表格,bjosn,详情见1.12、查看特定的数据,使用的时候shell命令,需要一些基础的语法知识,详情见2.13、更新数据,详情见3.14、删除数据,慎用,如果没有备份,删除…

    2022年8月21日
    5
  • 完全卸载flash浏览器插件_浏览器内置flash卸载

    完全卸载flash浏览器插件_浏览器内置flash卸载前天晚上不知道点到了什么东西,弹出来个窗口说浏览器的Flash插件需要升级,当时也没多想就确定了,结果发现QQ2009一开就崩溃,囧……找了半天发现问题出在升级的这个Flash插件上面,于是在控制面板中卸载掉,重新安装——一开QQ继续崩溃,继续囧……再查,发现在控制面板中卸载Flash插件之后,在C:\Windows\System32\Macromed\Flash\依旧存在两个文…

    2022年10月15日
    2
  • 字符串常量池概述[通俗易懂]

    字符串常量池概述[通俗易懂]字符串常量池概述常量池表(Constant_Pooltable)Class文件中存储所有常量(包括字符串)的table。这是Class文件中的内容,还不是运行时的内容,不要理解它是个池子,其实就是Class文件中的字节码指令。运行时常量池(RuntimeConstantPool)JVM内存中方法区的一部分,这是运行时的内容。这部分内容(绝大部分)是随着JVM运行时候,从常量池转化而来,每个Class对应一个运行时常量池。上一句中说绝大部分是因为:除了Class中常量池内容,还可能包括

    2022年7月28日
    11
  • IDEA汉化教程(一分钟即可)「建议收藏」

    IDEA汉化教程(一分钟即可)「建议收藏」Idea汉化教程(简单,一分钟即可)步骤打开idea点击左上角的File,然后点击Settings(如下图)进去后点击Plugins,然后点击Marketplace,然后再搜索框搜索chinese然后搜索出东西,点击下面标注的,点击安装然后下载好后再点击右侧的安装(由于我已经安装了,所以才有中文跟显示以安装)。安装完后重启就可以看到idea被汉化了…

    2022年5月11日
    95

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号