js的深淺拷貝可以分為數(shù)組的深淺拷貝和對(duì)象的深淺拷貝
一、數(shù)組的深淺拷貝
如果只是簡(jiǎn)單的將數(shù)組中的元素付給另外一個(gè)數(shù)組吼渡,那么只要新數(shù)組中的元素發(fā)生了改變容为,原數(shù)組中的元素也會(huì)相應(yīng)的發(fā)生改變,因?yàn)槿绻皇呛?jiǎn)單的賦值寺酪,那么它只是對(duì)引用了元素的地址坎背,并非真正意義上的拷貝,這種稱之為淺拷貝寄雀。
例如:
var arr = ['張三','李四','王五'];
var newArr = arr;
newArr[1] = '趙四';
console.log(arr); // ==> ['張三','趙四','王五']
console.log(newArr); // ==> ['張三','趙四','王五']
但是我們?cè)趹?yīng)用中往往不希望原數(shù)組的元素也被修改得滤,那么就需要深拷貝來(lái)解決這個(gè)問(wèn)題。
有兩種方法:
1盒犹、js的slice()方法
slice()slice() 方法可從已有的數(shù)組中返回選定的元素.
語(yǔ)法:arrayObject.slice(start,end);
start:必選懂更。規(guī)定從何處開始選取。如果是負(fù)數(shù)急膀,那么它規(guī)定從數(shù)組尾部開始算起的位置沮协。也就是說(shuō),-1 指最后一個(gè)元素卓嫂,-2 指倒數(shù)第二個(gè)元素慷暂,以此類推。
end:可選命黔。規(guī)定從何處結(jié)束選取呜呐。該參數(shù)是數(shù)組片斷結(jié)束處的數(shù)組下標(biāo)。如果沒有指定該參數(shù)悍募,那么切分的數(shù)組包含從 start 到數(shù)組結(jié)束的所有元素蘑辑。如果這個(gè)參數(shù)是負(fù)數(shù),那么它規(guī)定的是從數(shù)組尾部開始算起的元素坠宴。
返回值:返回一個(gè)新的數(shù)組洋魂,包含從 start 到 end (不包括該元素)的 arrayObject 中的元素。
var arr = ['張三','李四','王五'];
var newArr = arr.slice(0);
newArr[1] = '趙四';
console.log(arr); // ==> ['張三','李四','王五']
console.log(newArr); // ==> ['張三','趙四','王五']
2喜鼓、js的contat()方法
concat() 方法用于連接兩個(gè)或多個(gè)數(shù)組副砍。
該方法不會(huì)改變現(xiàn)有的數(shù)組,而僅僅會(huì)返回被連接數(shù)組的一個(gè)副本庄岖。
語(yǔ)法:arrayObject.concat(arrayX,arrayX,arrayX......)
返回值:返回一個(gè)新的數(shù)組豁翎。該數(shù)組是通過(guò)把所有 arrayX 參數(shù)添加到 arrayObject 中生成的。如果要進(jìn)行 concat() 操作的參數(shù)是數(shù)組隅忿,那么添加的是數(shù)組中的元素心剥,而不是數(shù)組。
var a = [1,2,3];
console.log(a.concat(4,5)); // ===>[1,2,3,4,5]
var newArr = [4,5];
console.log(arr.concat(newArr)); //===>[1,2,3,4,5]
var arr = ['張三','李四','王五'];
var newArr = arr.concat();
newArr[1] = '趙四';
console.log(arr); // ==> ['張三','李四','王五']
console.log(newArr); // ==> ['張三','趙四','王五']
二背桐、對(duì)象的深淺拷貝
例如:
var obj = {
'name' : 'zhangsan',
'hobby' : ['sport','singing','reading']
}
function copy(obj1) {
var obj2 = new Object();
for (var i in obj1) {
obj2[i] = obj1[i];
}
return obj2;
}
var newObj = copy(obj);
obj.hobby.push('hiking');
console.log(obj.hobby); // ['sport','singing','reading','hiking']
console.log(newObj.hobby) // ['sport','singing','reading','hiking']
可見淺拷貝只是對(duì)原對(duì)象進(jìn)行了數(shù)據(jù)的應(yīng)用优烧,子對(duì)象獲得的只是一個(gè)內(nèi)存地址,而不是真正拷貝链峭,因此存在父對(duì)象被篡改的可能畦娄。(對(duì)象中存在數(shù)組的時(shí)候)
function deepCopy(obj1, obj2) {
var obj2 = obj2 || {};
for (var i in obj1) {
if (typeof obj1[i] === 'object') {
obj2[i] = (obj1[i].constructor === Array) ? [] : {};
deepCopy(obj1[i], obj2[i]);
} else {
obj2[i] = obj1[i];
}
}
return obj2;
}
var obj = {
'name' : 'zhangsan',
'hobby' : ['sport','singing','reading'],
'friends' : ['lisi','wangwu']
}
var newObj = new Object();
deepCopy(obj,newObj);
obj.hobby.push('hiling');
console.log(newObj);
console.log(obj);