JS深拷贝与浅拷贝(附带通用深拷贝function方法)

所谓深拷贝与浅拷贝,乍一听我也蒙了,仔细一看其实就是C#中值类型复制和引用类型复制的区别。

值类型自然不存在这个概念,因为你怎么复制都是两个数据,但是引用类型就不一样了,引用类型涉及到引用的复制和值的复制问题。因此下面没有提到数据类型时候都是在说引用类型哈!

深拷贝就是引用类型复制时不仅仅将存在栈中的引用复制,同时还将堆中的数据复制了一份,从而得到了两份数据。

浅拷贝就是平常的赋值方式,仅仅将栈中的引用复制了,但是堆中的数据并没有复制。

【值类型数据放在栈中;引用类型数据在堆中,引用在栈里】


1.引用类型中常见的深拷贝方式

①数组:采用es6的...展开

    let a=[1,2,3],

    let b=[...a];

②json:采用es6的...展开

    let a={"c":12};

    let b={...a};

③系统对象

    Date等,直接将其实例作为新对象创建过程中构造函数的参数即可【因为系统对象已经帮你实现了这个过程】

    let a=new Date();

    let b=new Date(a);

    这样b就对a实现了深拷贝

④自定义对象

     与系统对象方式一样,但是前提你要自己写拷贝方法。

⑤symbol类型

    有关symbol内容请看:https://es6.ruanyifeng.com/#docs/symbol

     symbol类型的深拷贝要利用symbol.for进行,同时利用ES2019 提供了一个实例属性description,直接返回 Symbol 的描述。

      let s1=symbol.for('wayne');

      let s2=symbol.for(s1.description);

这样s2就实现了对s1的深拷贝。

2.完整深拷贝通用代码

        //[4]深拷贝方法
	function clone(data){
		//1.判断symbol类型
		if (typeof data=="symbol"){
			return symbol.for(data.description);
		}
		//2.判断值类型(即不是object类型)
		else if (typeof data!="object"){
			return data;
		}
		//3.判断数组
		else if (data.constructor===Array){
			return data.map(i=>clone(i));
		}
		//4.判断json类型
		else if (data.constructor===Object){
			let tem={};
			for(let key in data){
				tem[key]=clone(data[key]);
			}
			return tem;
		}
		//5.判断系统对象和自定义对象
		else{
			return new data.constructor(data);
		}
	}

注意,对于数组和json格式来说,单纯的...展开只能实现一层的拷贝,如果想多层拷贝,就要用到上述代码中3和4所写的那样,利用递归。

3.总结

目前针对深拷贝浅拷贝就这么多,欢迎留言指正讨论

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

韦_恩

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值