深浅拷贝知识在咱们的平常开发中还算是用的比较多,可是以前的状态一直都是只曾听闻,不曾使用(其实用了只是本身没有意识到),因此今天来跟你们聊一聊js的深浅拷贝;
javascript
首先咱们来了解一下javascript的数据类型,在ES5版本的js中咱们的javascript一共有6种数据类型,分别是:java
Number(数值型)、String(字符串)、Boolean(布尔型)、Object(对象,object和array都属于Object类型)、null、undefined数组
咱们平常使用的javascript深浅拷贝主要是面向Object引用类型进行拷贝;spa
咱们知道了js的深浅拷贝面对的执行操做对象,而后咱们再来看一下深浅拷贝的概念:指针
拷贝顾名思义就是复制,内存中一共分为栈内存和堆内存两大区域,所谓深浅拷贝主要是对javascript引用类型数据进行拷贝一份,浅拷贝就是引用类型数据相互赋值以后,例obj1=obj2;若是后面的操做中修改obj1或者obj2,这个时候数据是会进行相应的变化的,由于在内存中引用类型数据是存储在堆内存中,堆内存中存放的是引用类型的值,同时会有一个指针地址指向栈内存,两个引用类型数据地址同样,若是其中一个发生变化另一个都会有影响;而深拷贝则不会,深拷贝是会在堆内存中从新开辟一块空间进行存放;code
基本类型复制:对象
var a = 1; var b = a;//复制 console.log(b)//1 a = 2;//改变a的值 console.log(b)//1 console.log(a) //2
由于a,b都是属于基本类型,基本类型的复制是不会影响对方的,由于基本类型是每一次建立变量都会在栈内存中开辟一块内存,用来存放值,因此基本类型进行复制是不会对另一个变量有影响的;blog
引用类型复制:递归
引用类型的复制咱们分为数组的复制和对象的复制两个方面来进行讲解:ip
js的浅拷贝:
var arr1 = ['red','green']; var arr2 = arr1;//复制 console.log(arr2)//['red','green']; arr1.push('black') ;//改变color1的值 console.log(arr2)//['red','green','black'] console.log(arr1) //["red", "green", "black"]
上面的案例是javascript数组的浅拷贝,经过上面的知识咱们能够看知道数组是引用类型数据,引用类型数据复制是会进行相互影响的,咱们看到arr1.push('black')添加了一个新的子项,由于上面var arr2=arr1这行代码是将两个引用类型数据的地址指针指向了同一块堆内存区域,因此无论是arr1仍是arr2修改,任何一个一个改动两个数组都是会互相产生影响的;上面的那种直接赋值方式的复制就是咱们常说的引用类型的浅拷贝;
关于深拷贝不少同窗都误觉得js的原生方法concat、slice是属于深拷贝,其实不是的;js的原生方法concat、slice都是仅适用于一维数组,一旦到了二维数组或者多维数组中就会出现问题,就出现拷贝的不够完全致使仍是会发生数据的相互牵引问题;
slice:
var arr1 = ['red','green']; var arr2 = arr1.slice(0);//复制 console.log(arr2)//['red','green']; arr1.push('black') ;//改变color1的值 console.log(arr2)//["red", "green"] console.log(arr1)//["red", "green", "black"]
js原生的方法slice会返回一个新的数组,上述代码乍一看会觉得是深拷贝,由于arr2和arr1相互复制和牵引,而当arr1调用了push方法添加了新数组子项的时候,arr2没有发生变化;是的,这是符合深拷贝的特性,可是拷贝的不够完全,因此还不能算是真正意义上的深拷贝,因此slice只能被称为浅拷贝;slice方法只适用于一维数组的拷贝,在二维数组中就会破绽百出;
下面咱们再来看一下二维数组的例子:
var arr1=[1,2,3,['1','2','3']]; var arr2=arr1.slice(0); arr1[3][0]=0; console.log(arr1);//[1,2,3,['0','2','3']] console.log(arr2);//[1,2,3,['0','2','3']]
上述代码是一个二维数组,当咱们在arr1[3][0]里面进行更改arr1的值的时候,咱们发现arr一、arr2两个数组的值都发生了变化;因此事实证实slice不是深拷贝;
concat:
var arr1 = ['red','green']; var arr2 = arr1.concat();//复制 console.log(arr2)//['red','green']; arr1.push('black') ;//改变color1的值 console.log(arr2)//["red", "green"] console.log(arr1)//["red", "green", "black"]
var arr1=[1,2,3,['1','2','3']]; var arr2=arr1.concat(); arr1[3][0]=0; console.log(arr1);//[1,2,3,['0','2','3']] console.log(arr2);//[1,2,3,['0','2','3']]
concat方法在一维数组中是不会影响源数组的数据的,而在二维数组中concat的表现和slice是同样的;
js的深拷贝:
js数组中实现深拷贝的方法都多种,好比JSON.parse(JSON.stringify())和递归以及JQuery库的extend方法(只是extend方法须要依赖JQuery库,因此咱们尽可能的使用原生的方式来实现)都是能够实现数组和对象的深拷贝的;
var arr1 = ['red','green']; var arr2 = JSON.parse(JSON.stringify(arr1));//复制 console.log(arr2)//['red','green']; arr1.push('black') ;//改变color1的值 console.log(arr2)//["red", "green"] console.log(arr1)//["red", "green", "black"]
上述代码中咱们能够清晰的看到JSON.parse(JSON.stringify())是真正意义上实现了深拷贝;
递归实现深拷贝:
function deepClone(obj){ //判断参数是否是一个对象 let objClone = obj instanceof Object?[]:{}; 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; } var a ={ x:1, y:2 }; b=deepClone(a); a.x=3 console.log(a); console.log(b);
输出效果以下:
总结:
1:深拷贝只是从源数据中拷贝一份出来进行操做,而不是改变源数据;改变源数据的那是浅拷贝;
2:原生js方法slice、concat都不是真正意义上的深拷贝,都仅只适用于一维数组,拷贝的属性不够完全;
3:实现js深拷贝咱们能够经过JSON.parse(JSON.stringify())、递归以及JQuery库的extend方法来实现;