深拷贝与浅拷贝
前言
JS数据类型
- 基本数据类型:String、Boolean、Number、Undefined、Null
- 引用数据类型:Object(Array、Date、RegExp、Function)
区别
保存位置不同:基本数据类型,名字和值(键值)都存储在栈内存中;引用数据类型,名字存在栈内存中,值存在堆内存中,但栈内存会提供一个引用的地址指向堆内存的值。
原因:
- 堆比栈大,栈比堆速度快;
- 基本数据类型稳定,相对来说占用的内存小;
- 引用类型数据大小是动态且无限的,所以将值存在堆中;
- 堆内存是无序存储,可以根据引用直接获取
因此,深拷贝与浅拷贝只针对于引用数据类型来说的。
定义
浅拷贝:只复制指向某个对象的指针,而不复制对象本身,新旧对象还是共享同一块内存。
深拷贝:会另外创造一个一摸一样的对象,新对象和原对象不共享内存,修改新对象不会改变原对象。
区别:浅拷贝只复制对象的第一层属性、深拷贝可以对对象的属性进行递归复制
浅拷贝的实现
for···in只循环第一层
// 只复制第一层的浅拷贝function simpleCopy(obj1) { var obj2 = Array.isArray(obj1) ? [] : {}; for (let i in obj1) { obj2[i] = obj1[i]; } return obj2;}var obj1 = { a: 1, b: 2, c: { d: 3 }}var obj2 = simpleCopy(obj1);obj2.a = 3;obj2.c.d = 4;alert(obj1.a); // 1alert(obj2.a); // 3alert(obj1.c.d); // 4alert(obj2.c.d); // 4
Object.assign()
var obj = { a: 1, b: 2}var obj1 = Object.assign({},obj);boj1.a = 3;console.log(obj.a) // 3
直接赋值
let a=[0,1,2,3,4], b=a;console.log(a===b);a[0]=1;console.log(a,b);
深拷贝的实现
采用递归去拷贝所有层级属性
function deepClone(obj){ let objClone = Array.isArray(obj)?[]:{}; if(obj && typeof obj==="object"){ for(key in obj){ if(obj.hasOwnProperty(key)){ //判断ojb子元素是否为对象,如果是,递归复制 if(obj[key]&&typeof obj[key] ==="object"){ objClone[key] = deepClone(obj[key]); }else{ //如果不是,简单复制 objClone[key] = obj[key]; } } } } return objClone;} let a=[1,2,3,4], b=deepClone(a);a[0]=2;console.log(a,b);
缺陷:当遇到两个互相引用的对象,会出现死循环,为了避免相互引用的对象导致死循环,应该在遍历的时候判断是否相互引用对象,如果是则退出循环
通过JSON对象来实现拷贝(无法实现对对象中方法的深拷贝,会显示为undefined)
function deepClone2(obj) { var _obj = JSON.stringify(obj), objClone = JSON.parse(_obj); return objClone;}
通过jQuery的extend方法实现深拷贝
var array = [1,2,3,4];var newArray = $.extend(true,[],array); // true为深拷贝,false为浅拷贝
lodash函数库实现深拷贝
let result = _.cloneDeep(test)
Reflect法
// 代理法function deepClone(obj) { if (!isObject(obj)) { throw new Error('obj 不是一个对象!') } let isArray = Array.isArray(obj) let cloneObj = isArray ? [...obj] : { ...obj } Reflect.ownKeys(cloneObj).forEach(key => { cloneObj[key] = isObject(obj[key]) ? deepClone(obj[key]) : obj[key] }) return cloneObj}