在 JS 中有一些基本类型像是Number
、String
、Boolean,
而对象就是像这样的东西{ name: 'Larry', skill: 'Node.js' },对象跟基本类型最大的不一样就在于他们的传值方式。javascript
基本类型是按值传递,像是这样:在修改a
时并不会改到b
java
var a = 25; var b = a; b = 18; console.log(a);//25 console.log(b);//18
但对象就不一样,对象传的是按引用传值:jquery
var obj1 = { a: 10, b: 20, c: 30 }; var obj2 = obj1; obj2.b = 100; console.log(obj1); // { a: 10, b: 100, c: 30 } <-- b 被改到了 console.log(obj2); // { a: 10, b: 100, c: 30 }
复制一份obj1
叫作obj2,
而后把obj2.b
改为100,
但却不当心改到obj1.b,
由于他们根本是同一个对象,这就是所谓的浅拷贝。json
要避免这样的错误发生就要写成这样:数据结构
var obj1 = { a: 10, b: 20, c: 30 }; var obj2 = { a: obj1.a, b: obj1.b, c: obj1.c }; obj2.b = 100; console.log(obj1); // { a: 10, b: 20, c: 30 } <-- b 沒被改到 console.log(obj2); // { a: 10, b: 100, c: 30 }
这样就是深拷贝,不会改到本来的obj1。
函数
浅拷贝只复制指向某个对象的指针,而不复制对象自己,新旧对象仍是共享同一块内存。但深拷贝会另外创造一个如出一辙的对象,新对象跟原对象不共享内存,修改新对象不会改到原对象。性能
也就是简单地复制而已ui
一、简单地复制语句spa
<script type="text/javascript"> function simpleClone(initalObj) { var obj = {}; for ( var i in initalObj) { obj[i] = initalObj[i]; } return obj; } var obj = { a: "hello", b:{ a: "world", b: 21 }, c:["Bob", "Tom", "Jenny"], d:function() { alert("hello world"); } } var cloneObj = simpleClone(obj); console.log(cloneObj.b); console.log(cloneObj.c); console.log(cloneObj.d); cloneObj.b.a = "changed"; cloneObj.c = [1, 2, 3]; cloneObj.d = function() { alert("changed"); }; console.log(obj.b); console.log(obj.c); console.log(obj.d); </script>
结果为:指针
二、Object.assign()
方法能够把任意多个的源对象自身的可枚举属性拷贝给目标对象,而后返回目标对象。可是 Object.assign
是ES6的新函数。Object.assign()Object.assign()
进行的是浅拷贝,拷贝的是对象的属性的引用,而不是对象自己。
Object.assign(target, ...sources)
参数:
target:目标对象。
sources:任意多个源对象。
返回值:目标对象会被返回。
var obj = { a: {a: "hello", b: 21} }; var initalObj = Object.assign({}, obj); initalObj.a.a = "changed"; console.log(obj.a.a); // "changed"
兼容性:
须要注意的是:
Object.assign()能够处理一层的深度拷贝,以下:
var obj1 = { a: 10, b: 20, c: 30 }; var obj2 = Object.assign({}, obj1); obj2.b = 100; console.log(obj1); // { a: 10, b: 20, c: 30 } <-- 沒被改到 console.log(obj2); // { a: 10, b: 100, c: 30 }
要彻底复制又不能修改到原对象,这时候就要用 Deep Copy,这里会介绍几种Deep Copy 的方式。
一、手动复制
把一个对象的属性复制给另外一个对象的属性
var obj1 = { a: 10, b: 20, c: 30 }; var obj2 = { a: obj1.a, b: obj1.b, c: obj1.c }; obj2.b = 100; console.log(obj1); // { a: 10, b: 20, c: 30 } <-- 沒被改到 console.log(obj2); // { a: 10, b: 100, c: 30 }
但这样很麻烦,要一个一个本身复制;并且这样的本质也不能算是 Deep Copy,由于对象里面也可能回事对象,如像下面这个情况:
var obj1 = { body: { a: 10 } }; var obj2 = { body: obj1.body }; obj2.body.a = 20; console.log(obj1); // { body: { a: 20 } } <-- 被改到了 console.log(obj2); // { body: { a: 20 } } console.log(obj1 === obj2); // false console.log(obj1.body === obj2.body); // true
虽然obj1
跟obj2
是不一样对象,但他们会共享同一个obj1.body
,
因此修改obj2.body.a
时也会修改到旧的。
二、对象只有一层的话可使用上面的:Object.assign()函数
Object.assign({}, obj1)
的意思是先创建一个空对象{},接着把obj1
中全部的属性复制过去,因此obj2
会长得跟obj1
同样,这时候再修改obj2.b
也不会影响obj1。
由于Object.assign
跟咱们手动复制的效果相同,因此同样只能处理深度只有一层的对象,没办法作到真正的 Deep Copy。不过若是要复制的对象只有一层的话能够考虑使用它。
三、转成 JSON 再转回来
用JSON.stringify
把对象转成字符串,再用JSON.parse
把字符串转成新的对象。
var obj1 = { body: { a: 10 } }; var obj2 = JSON.parse(JSON.stringify(obj1)); obj2.body.a = 20; console.log(obj1); // { body: { a: 10 } } <-- 沒被改到 console.log(obj2); // { body: { a: 20 } } console.log(obj1 === obj2); // false console.log(obj1.body === obj2.body); // false
这样作是真正的Deep Copy,这种方法简单易用。
可是这种方法也有很多坏处,譬如它会抛弃对象的constructor。也就是深拷贝以后,无论这个对象原来的构造函数是什么,在深拷贝以后都会变成Object。
这种方法能正确处理的对象只有 Number, String, Boolean, Array, 扁平对象
,即那些可以被 json 直接表示的数据结构。RegExp对象是没法经过这种方式深拷贝。
也就是说,只有能够转成JSON
格式的对象才能够这样用,像function
没办法转成JSON。
var obj1 = { fun: function(){ console.log(123) } }; var obj2 = JSON.parse(JSON.stringify(obj1)); console.log(typeof obj1.fun); // 'function' console.log(typeof obj2.fun); // 'undefined' <-- 没复制
要复制的function
会直接消失,因此这个方法只能用在单纯只有数据的对象。
四、递归拷贝
function deepClone(initalObj, finalObj) { var obj = finalObj || {}; for (var i in initalObj) { if (typeof initalObj[i] === 'object') { obj[i] = (initalObj[i].constructor === Array) ? [] : {}; arguments.callee(initalObj[i], obj[i]); } else { obj[i] = initalObj[i]; } } return obj; } var str = {}; var obj = { a: {a: "hello", b: 21} }; deepClone(obj, str); console.log(str.a);
上述代码确实能够实现深拷贝。可是当遇到两个互相引用的对象,会出现死循环的状况。
为了不相互引用的对象致使死循环的状况,则应该在遍历的时候判断是否相互引用对象,若是是则退出循环。
改进版代码以下:
function deepClone(initalObj, finalObj) { var obj = finalObj || {}; for (var i in initalObj) { var prop = initalObj[i]; // 避免相互引用对象致使死循环,如initalObj.a = initalObj的状况 if(prop === obj) { continue; } if (typeof prop === 'object') { obj[i] = (prop.constructor === Array) ? [] : {}; arguments.callee(prop, obj[i]); } else { obj[i] = prop; } } return obj; } var str = {}; var obj = { a: {a: "hello", b: 21} }; deepClone(obj, str); console.log(str.a);
五、使用Object.create()方法
直接使用var newObj = Object.create(oldObj),能够达到深拷贝的效果。
function deepClone(initalObj, finalObj) { var obj = finalObj || {}; for (var i in initalObj) { var prop = initalObj[i]; // 避免相互引用对象致使死循环,如initalObj.a = initalObj的状况 if(prop === obj) { continue; } if (typeof prop === 'object') { obj[i] = (prop.constructor === Array) ? [] : Object.create(prop); } else { obj[i] = prop; } } return obj; }
六、jquery
jquery 有提供一个$.extend
能够用来作 Deep Copy。
var $ = require('jquery'); var obj1 = { a: 1, b: { f: { g: 1 } }, c: [1, 2, 3] }; var obj2 = $.extend(true, {}, obj1); console.log(obj1.b.f === obj2.b.f); // false
七、lodash
另一个很热门的函数库lodash,也有提供_.cloneDeep
用来作 Deep Copy。
var _ = require('lodash'); var obj1 = { a: 1, b: { f: { g: 1 } }, c: [1, 2, 3] }; var obj2 = _.cloneDeep(obj1); console.log(obj1.b.f === obj2.b.f); // false
这个性能还不错,使用起来也很简单。