JS中实现浅拷贝和深拷贝的代码详解
(一)JS中基本类型和引用类型
JavaScript的变量中包含两种类型的值基本类型值 和 引用类型值,在内存中的表现形式在于前者是存储在栈中的一些简单的数据段,后者则是保存在堆内存中的一个对象。
基本类型值
在JavaScript中基本数据类型有 String , Number , Undefined , Null , Boolean ,在ES6中,又定义了一种新的基本数据类型 Symbol ,所以一共有6种。
基本类型是按值访问的,从一个变量复制基本类型的值到另一个变量后,这两个变量的值是完全独立的,即使一个变量改变了也不会影响到第二个变量。
var str1 = '撩课'; var str2 = str1; str2 = 'itlike'; console.log(str2); //'itlike' console.log(str1); //'撩课'
引用类型值
引用类型值是引用类型的实例,它是保存在堆内存中的一个对象,引用类型是一种数据结构,最常用的是Object,Array,Function类型,还有Date,RegExp,Error等。
在ES6中提供了Set,Map2种新的数据结构。
(二)JS中如何复制引用类型的
基本类型和引用类型赋值的差异化
举个例子在狼蚁网站SEO优化代码中,只修改了obj1中的name属性,却改变了ob1和obj2中的name属性。
var obj1 = {'name': '撩课'}; var obj2 = obj1; obj2.name = '小撩'; console.log(obj1); // {'name': '撩课'} console.log(obj2); // {'name': '撩课'}
当变量复制引用类型值的时候,同样和基本类型值一样会将变量的值复制到新变量上,不同的是对于变量的值,它是一个指针,指向存储在堆内存中的对象。
因为,在JS中,堆内存中的对象无法直接访问,必须要访问这个对象在堆内存中的地址,然后再按照这个地址去获得这个对象中的值。
(三)浅拷贝
在JS中,如果属性是基本类型,拷贝的就是基本类型的值;如果属性是引用类型,拷贝的就是内存地址;所以如果其中一个对象改变了这个地址,就会影响到另一个对象。
狼蚁网站SEO优化是JavaScript提供的浅拷贝方法
Object.assign
ES6中拷贝对象的方法,接受的第一个参数是拷贝的目标,剩下的参数是拷贝的源对象;
语法Object.assign(target, ...sources)
var p = { 'name': '张三', }; var copyP = {}; Object.assign(copyP, p); console.log(copyP); console.log(p);
Object.assign是一个浅拷贝,它只是在根属性(对象的第一层级)创建了一个新的对象,如果属性的值是对象的话,只会拷贝一份相同的内存地址。
扩展运算符
利用扩展运算符可以在构造字面量对象时,进行克隆或者属性拷贝。语法如下
var cloneObj = { ...obj }; var obj = {'name': '撩课', 'college': ['H5','JAVA','Python']} var obj2 = {...obj}; obj.name='小撩'; //{'name': '小撩', 'college': ['H5','JAVA','Python']} console.log(obj); //{'name': '撩课', 'college': ['H5','JAVA','Python']} console.log(obj2); obj.college.push('Go'); //{'name': '小撩', 'college': ['H5','JAVA','Python','Go']} console.log(obj); //{'name': '小撩', 'college': ['H5','JAVA','Python','Go']} console.log(obj2);
扩展运算符和Object.assign()存在同样的问题,对于值是对象的属性无法完全拷贝成两个不同对象;
如果属性都是基本类型的值的话,使用扩展运算符更加简洁。
(四)深拷贝
浅拷贝只在根属性上在堆内存中创建了一个新的的对象,复制了基本类型的值,复杂数据类型也就是对象则是拷贝相同的地址。
而深拷贝则是将一个对象从内存中完整的拷贝一份出来,从堆内存中开辟一个新的区域存放新对象,且修改新对象不会影响原对象。
JSON.stringify
JSON.stringify()是目前开发过程中最常用的深拷贝方式,原理是把一个对象序列化成为一个JSON字符串,将对象的内容转换成字符串的形式再保存在内存中,再用JSON.parse()反序列化将JSON字符串变成一个新的对象。
举个例子
var obj = { name: '撩课', age: 18, friends: ['小花', '小黑'], goodF: { name: '小撩', age: 19, address: '上海', pets: [{name: '土豆'}, {name: '马铃薯'}]}, bir: new Date() }; var newObj = JSON.parse(JSON.stringify(obj)); obj.goodF.pets[0].name = '旺财'; console.log(newObj); console.log(obj);
使用JSON.stringify实现深拷贝有几点要注意
1)拷贝的对象的值中如果有函数,undefined,symbol,经过JSON.stringify()序列化后的JSON字符串中这个键值对会消失;
无法拷贝不可枚举的属性,无法拷贝对象的原型链
3)拷贝Date引用类型会变成字符串
4)拷贝RegExp引用类型会变成空对象
对象中含有NaN、Infinity和-Infinity,则序列化的结果会变成null
递归实现深拷贝
具体实现如下
/ 辅助函数, 判定是否是对象 @param obj @returns {boolean} / function isObj(obj) { return obj instanceof Object; } / 深拷贝fromObj面的所有属性/值, 到toObj对象里面 @param fromObj 拷贝对象 @param toObj 目标对象 / function deepCopyObj2NewObj(fromObj, toObj) { for (var key in fromObj) { if(fromObj.hasOwnProperty(key)){ var fromValue = fromObj[key]; // 如果是值类型,那么就直接拷贝赋值 if (!isObj(fromValue)) { toObj[key] = fromValue; } else { // 如果是引用类型,那么就再调用一次这个方法, // 去内部拷贝这个对象的所有属性 // fromValue是什么类型, 创建一个该类型的空对象 var tmpObj = new fromValue.constructor; // console.log(tmpObj); // debugger; deepCopyObj2NewObj(fromValue, tmpObj); toObj[key] = tmpObj; } } } }
(五)
1)在日常开发中一般并不需要拷贝很多特殊的引用类型,深拷贝对象使用JSON.stringify
是最直接和简单的方法。
2)实现一个完整的深拷贝是非常复杂的,需要考虑到很多边界情况。对于特殊的引用类型有拷贝需求的话,建议借助第三方完整的库。
以上所述是长沙网络推广给大家介绍的JS中实现浅拷贝和深拷贝的代码详解,希望对大家有所帮助,如果大家有任何疑问请给我留言,长沙网络推广会及时回复大家的。在此也非常感谢大家对狼蚁SEO网站的支持!
如果你觉得本文对你有帮助,欢迎网络推广网站推广转载,烦请注明出处,谢谢!
编程语言
- 如何快速学会编程 如何快速学会ug编程
- 免费学编程的app 推荐12个免费学编程的好网站
- 电脑怎么编程:电脑怎么编程网咯游戏菜单图标
- 如何写代码新手教学 如何写代码新手教学手机
- 基础编程入门教程视频 基础编程入门教程视频华
- 编程演示:编程演示浦丰投针过程
- 乐高编程加盟 乐高积木编程加盟
- 跟我学plc编程 plc编程自学入门视频教程
- ug编程成航林总 ug编程实战视频
- 孩子学编程的好处和坏处
- 初学者学编程该从哪里开始 新手学编程从哪里入
- 慢走丝编程 慢走丝编程难学吗
- 国内十强少儿编程机构 中国少儿编程机构十强有
- 成人计算机速成培训班 成人计算机速成培训班办
- 孩子学编程网上课程哪家好 儿童学编程比较好的
- 代码编程教学入门软件 代码编程教程