JS深拷贝(deepClone)比较实用的两种方法

分享下js深拷贝对象,两种最常用、最实用的方法

方法1、递归遍历对象的属性,返回一个新的对象
// 这里有两种写法,都可用

const obj1 = {
	name: 'wang',
	f: function() {},
	tar: {
		name:'Jason',
		age: 18
	}
}
function deepClone(obj) {
// 写法1
	// let newObj = obj.constructor === Array ? [] : {}
	// if(typeof obj !== 'object') {
	// 	return obj
	// }else{
	// 	for(var i in obj) {
	// 		if(typeof obj[i] === 'object'){
	// 			newObj[i] = deepClone(obj[i])
	// 		}else{
	// 			newObj[i] = obj[i]
	// 		}
	// 	}
	// }
	// return newObj;
	
// 写法2
  if (!obj && typeof obj !== "object") {
    throw new Error("error arguments", "deepClone");
  }
  const targetObj = obj.constructor === Array ? [] : {};
  	Object.keys(obj).forEach(keys => {
    if (obj[keys] && typeof obj[keys] === "object") {
      targetObj[keys] = deepClone(obj[keys]);
    } else {
      targetObj[keys] = obj[keys];
    }
  });
  return targetObj;
}
var obj2 = deepClone(obj1)
console.log(obj2)

方法2:通过Object.create和Object.defineProperty取对象的所有属性,返回一个新的对象
// 方法2和方法1的区别在于,使用方法1不能拷贝对象中的时间和正则

var obj = {
  name: 'wang',
  f: function() {},
  tar: {
	name:'Jason',
	age: 18,
	cc: {
		lang: 'typescript'
	}
  },
  arr: ['11', '22'],
  reg: /\s|\n/g,
  date: new Date()
}
function deepClone(orig) {
	var copy = Object.create(Object.getPrototypeOf(orig))
	copyOwnPropertiesFrom(copy,orig);
	return copy;
}

function copyOwnPropertiesFrom(target, source) {
	Object.getOwnPropertyNames(source)
		.forEach(ele =>{
			// 获取到每个属性的所有描述,然后重新定义对象的属性并返回此对象。
			var desc = Object.getOwnPropertyDescriptor(source,ele);
			Object.defineProperty(target,ele,desc)
		});
	return target;
}
deepClone(obj)
深拷贝的方法有很多种,这里只提供两种最常用的,主要是看哪种最适用你自己
  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
深拷贝是指完全复制一个对象,包括它的引用类型数据(如数组、对象等),而不是只复制它们的地址或引用。深拷贝的目的是为了保持原始对象和新对象的独立性,避免它们之间相互影响。 实现深拷贝方法有很多,比较常用的有递归、JSON序列化和反序列化、lodash库等。下面分别介绍这些方法的实现方式。 1. 递归 使用递归的方式可以遍历对象的每一个属性,并将其复制到新的对象中。同时,如果属性的值仍然是一个对象,就递归地调用深拷贝函数,直到遍历完所有的属性。 ```javascript function deepClone(obj) { if (typeof obj !== 'object' || obj === null) { return obj; } const newObj = Array.isArray(obj) ? [] : {}; for (const key in obj) { if (Object.hasOwnProperty.call(obj, key)) { newObj[key] = deepClone(obj[key]); } } return newObj; } ``` 2. JSON序列化和反序列化 使用JSON.stringify()方法将原始对象序列化为一个JSON字符串,然后使用JSON.parse()方法JSON字符串反序列化为一个新的对象。 ```javascript function deepClone(obj) { return JSON.parse(JSON.stringify(obj)); } ``` 这种方法的缺点是,如果原始对象中有函数、正则表达式、Date等对象时,在序列化和反序列化过程中会丢失它们的类型,变成了字符串或空对象。 3. lodash库 lodash是一个常用的JavaScript工具库,提供了许多方便的函数,包括深拷贝函数。 ```javascript const _ = require('lodash'); const obj = {a: 1, b: {c: 2, d: {e: 3}}}; const newObj = _.cloneDeep(obj); ``` 使用lodash库的深拷贝函数可以处理原始对象中的函数、正则表达式、Date等对象,保持它们的类型不变。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值