javascript --- > 实现对象的深拷贝

107 篇文章 1 订阅
49 篇文章 0 订阅

浅拷贝和深拷贝

  • 浅拷贝: 只拷贝一层.当对象是复杂数据类型(Object、 Array)时,只拷贝引用
  • 深拷贝: 多层拷贝.复杂数据类型,会重新分配内存空间.

实现浅拷贝的2种方法

  • 使用for ... in 实现
var obj = {
	name: 'marron',
	age: 18,
	msg: {
		sex: "1" 
	}
}
var o = {};
for(let k in obj) {
	o[k] = obj[k]
};
console.log(o);
  • 使用es6提供的方法Object.assign
var obj = {
	name: 'marron',
	age: 18,
	msg: {
		sex: "1"
	}
}
var o = {};
Object.assign(o, obj);
console.log(o);

上面用实现了将对象obj赋值给o. 但是,o中对属性sex的操作.是引用操作.
即改变o.msg.sex属性obj.msg.sex属性也会改变.

o.msg.sex = 2;
console.log(obj.msg.sex);	//	2

实现深拷贝

很明显上面并不是我们所需要的结果.
核心思路是,复杂数据类型赋值时先开辟内存空间.
尝试使用递归实现拷贝,思路如下:

  1. 使用for...in进行赋值
  2. 在赋值时首先判断当前的属性是否为数组a instanceof Array,若为数组,则按照数组方式进行递归.
  3. 在判断当前的属性是否为对象a instanceof Object
  4. 剩下的认为是简单数据类型…直接赋值即可
const deepCopy = (newobj, oldobj)  => {
	for(let k in oldobj) {
		let item = oldobj[k];
		if(item instanceof Array) {
			newobj[k] = [];
			deepCopy(newobj[k], oldobj[k])
		} else if (item instanceof Object){
			newobj[k] = {};
			deepCopy(newobj[k], oldobj[k])
		} else {
			newobj[k] = item
		}
	}
}
  • 测试:
let obj = {
	id: 1,
	name: 'marron',
	msg: {
		age: 18
	}
}
let o = deepCopy(o, obj);
console.log(o);
o.msg.age = 20;
console.log(obj);

在这里插入图片描述

稍微改进一下

  • 上面传递了2个参数,且没有考虑日期和正则的情况
  • 改变递归出来的条件
  • 使用typeof判断是否是 数组和对象的形式
  • 但是使用 typeof === ‘object’ 会遗漏掉掉null的情况
  • 然后使用instanceof来判断正则和日期的形式
const deepClone = (obj) =>{
	if(obj === null) return null
	if(typeof obj !== 'object') return obj
	if(obj instanceof RegExp){
		return new RegExp(obj)
	} 
	if(obj instanceof Date) {
		return new Date(obj);
	}
	let newObj = new obj.constructor;
	for(let k in obj){
		if(obj.hasOwnProperty(k)){
			newObj[k] = deepClone(obj[k])
		}
	}
	return newObj
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值