JavaScript 數(shù)組的深度復(fù)制解析
對(duì)于javascript而言,數(shù)組是引用類型,如果要想復(fù)制一個(gè)數(shù)組就要?jiǎng)幽X袋想想了,因?yàn)榘╟oncat、slice在內(nèi)的函數(shù),都是淺層復(fù)制。也就是說,對(duì)于一個(gè)二維數(shù)組來說,用concat來做復(fù)制,第二維的數(shù)組還是引用,修改了新數(shù)組同樣會(huì)使舊數(shù)組發(fā)生改變。
于是乎,想要寫一個(gè)深度復(fù)制的函數(shù),來幫助做組數(shù)的深度復(fù)制。
一般情況下,使用 “=” 可以實(shí)現(xiàn)賦值。但對(duì)于數(shù)組、對(duì)象、函數(shù)等這些引用類型的數(shù)據(jù),這個(gè)符號(hào)就不好使了。
1. 數(shù)組的簡(jiǎn)單復(fù)制
1.1 簡(jiǎn)單遍歷
最簡(jiǎn)單也最基礎(chǔ)的方式,自然是循環(huán)處理。示例:
JavaScript
function array_copy(arr) { var out = [], i, len; if (out[i] instanceof Array === false){ return arr; } for (i = 0, len = arr.length; i < len; i++) { if (out[i] instanceof Array){ out[i] = deepcopy(arr[i]); } else { out[i] = arr[i]; } }; return a; } //測(cè)試 var arr1 = [1, 2, 3, 4], arr2 = array_copy(arr1); console.log(arr1, arr2); arr2[2] = 10; console.log(arr1[2], arr2[2]); function array_copy(arr) { var out = [], i, len; if (out[i] instanceof Array === false){ return arr; } for (i = 0, len = arr.length; i < len; i++) { if (out[i] instanceof Array){ out[i] = deepcopy(arr[i]); } else { out[i] = arr[i]; } }; return a; } //測(cè)試 var arr1 = [1, 2, 3, 4], arr2 = array_copy(arr1); console.log(arr1, arr2); arr2[2] = 10; console.log(arr1[2], arr2[2]);
1.2 變通的復(fù)制實(shí)現(xiàn)
經(jīng)常出現(xiàn)在面試題中的取巧方法,是使用 slice 或 contcat 方法實(shí)現(xiàn)。示例:
JavaScript
var arr1 = [1, 2, 3, 4], arr2 = arr1.slice(0), arr3 = arr1.concat(); console.log(arr1, arr2, arr3); arr2[2] = 10; arr3[2] = 11; console.log(arr1[2], arr2[2], arr3[2]); var arr1 = [1, 2, 3, 4], arr2 = arr1.slice(0), arr3 = arr1.concat(); console.log(arr1, arr2, arr3); arr2[2] = 10; arr3[2] = 11; console.log(arr1[2], arr2[2], arr3[2]);
2. 數(shù)組的深度復(fù)制
普通的一維數(shù)組且值為非引用類型,使用上述方法是沒有問題的,否則就比較麻煩了。深度復(fù)制需要考慮數(shù)組值為各種引用類型的情況。
2.1 使用 JSON 方法
JSON.stringify(array) 然后再 JSON.parse()。示例:
JavaScript
var arr1 = [1, 2, [3, 4], {a: 5, b: 6}, 7], arr2 = JSON.parse(JSON.stringify(arr1)); console.log(arr1, arr2); arr2[1] = 10; arr2[3].a = 20; console.log(arr1[1], arr2[1]); console.log(arr1[3], arr2[3]); var arr1 = [1, 2, [3, 4], {a: 5, b: 6}, 7], arr2 = JSON.parse(JSON.stringify(arr1)); console.log(arr1, arr2); arr2[1] = 10; arr2[3].a = 20; console.log(arr1[1], arr2[1]); console.log(arr1[3], arr2[3]);
此方法存在對(duì)古老瀏覽器的兼容性問題。如確需要作兼容,可引入如下兼容文件解決:
https://github.com/douglascrockford/JSON-js/blob/master/json2.js
注意:如果數(shù)組值為函數(shù),上述方法還是不行的。
2.2 深度復(fù)制的完全實(shí)現(xiàn)
考慮到多維數(shù)組的嵌套,以及數(shù)組值為對(duì)象的情況,可以作如下原型擴(kuò)展(當(dāng)然寫為普通函數(shù)實(shí)現(xiàn)也是可以的,原型擴(kuò)展是不建議的方式):
JavaScript
Object.prototype.clone = function () { var o = {}; for (var i in this) { o[i] = this[i]; } return o; }; Array.prototype.clone = function () { var arr = []; for (var i = 0; i < this.length; i++) if (typeof this[i] !== 'object') { arr.push(this[i]); } else { arr.push(this[i].clone()); } return arr; }; //測(cè)試1 Object var obj1 = { name: 'Rattz', age: 20, hello: function () { return "I'm " + name; } }; var obj2 = obj1.clone(); obj2.age++; console.log(obj1.age); //測(cè)試2 Array var fun = function(log) {console.log}, arr1 = [1, 2, [3, 4], {a: 5, b: 6}, fun], arr2 = arr1.clone(); console.log(arr1, arr2); arr2[2][1]= 'Mike'; arr2[3].a = 50; arr2[4] = 10; console.log(arr1, arr2); Object.prototype.clone = function () { var o = {}; for (var i in this) { o[i] = this[i]; } return o; Array.prototype.clone = function () { var arr = []; for (var i = 0; i < this.length; i++) if (typeof this[i] !== 'object') { arr.push(this[i]); } else { arr.push(this[i].clone()); } return arr; //測(cè)試1 Object var obj1 = { name: 'Rattz', age: 20, hello: function () { return "I'm " + name; } var obj2 = obj1.clone(); console.log(obj1.age); //測(cè)試2 Array var fun = function(log) {console.log}, arr1 = [1, 2, [3, 4], {a: 5, b: 6}, fun], arr2 = arr1.clone(); console.log(arr1, arr2); arr2[2][1]= 'Mike'; arr2[3].a = 50; arr2[4] = 10; console.log(arr1, arr2);
2.3 使用 jQuery 的 extend 方法
如果你在使用 jQuery,那么最簡(jiǎn)單的方法是使用 extend 插件方法。示例:
JavaScript
var arr1 = [1, 2, [3, 4], {a: 5, b: 6}, 7], arr2 = $.extend(true, [], arr1); console.log(arr1, arr2); arr2[1] = 10; console.log(arr1, arr2); var arr1 = [1, 2, [3, 4], {a: 5, b: 6}, 7], arr2 = $.extend(true, [], arr1); console.log(arr1, arr2); arr2[1] = 10; console.log(arr1, arr2);
以上所述是小編給大家介紹的JavaScript 數(shù)組的深度復(fù)制解析,希望對(duì)大家有所幫助,如果大家有任何疑問請(qǐng)給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對(duì)腳本之家網(wǎng)站的支持!
- jQuery中json對(duì)象的復(fù)制方式介紹(數(shù)組及對(duì)象)
- JavaScript數(shù)組復(fù)制詳解
- javascript 三種數(shù)組復(fù)制方法的性能對(duì)比
- Javascript 復(fù)制數(shù)組實(shí)現(xiàn)代碼
- javascript復(fù)制對(duì)象使用說明
- js中如何復(fù)制一個(gè)對(duì)象并獲取其所有屬性和屬性對(duì)應(yīng)的值
- 原生js實(shí)現(xiàn)復(fù)制對(duì)象、擴(kuò)展對(duì)象 類似jquery中的extend()方法
- 深入理解JavaScript中的對(duì)象復(fù)制(Object Clone)
- 改進(jìn)版通過Json對(duì)象實(shí)現(xiàn)深復(fù)制的方法
- JavaScript數(shù)組和對(duì)象的復(fù)制
相關(guān)文章
Javascript對(duì)象Clone實(shí)例分析
這篇文章主要介紹了Javascript對(duì)象Clone用法,實(shí)例分析了javascript對(duì)象克隆的相關(guān)技巧,需要的朋友可以參考下2015-06-06JavaScript實(shí)現(xiàn)京東快遞單號(hào)查詢
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)京東快遞單號(hào)查詢,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-11-11javascript產(chǎn)生隨機(jī)數(shù)方法匯總
這篇文章主要介紹了javascript產(chǎn)生隨機(jī)數(shù)方法匯總的相關(guān)資料,需要的朋友可以參考下2016-01-01關(guān)于Javascript加載執(zhí)行優(yōu)化的研究報(bào)告
這篇文章主要介紹了關(guān)于Javascript加載執(zhí)行優(yōu)化的研究報(bào)告,需要的朋友可以參考下2014-12-12