JavaScript 中物件的深拷貝
在JavaScript中,對物件進行拷貝的場景比較常見。但是簡單的複製語句只能對物件進行淺拷貝,即複製的是一份引用,而不是它所引用的物件。而更多的時候,我們希望對物件進行深拷貝,避免原始物件被無意修改。
物件的深拷貝與淺拷貝的區別如下:
- 淺拷貝:僅僅複製物件的引用,而不是物件本身;
- 深拷貝:把複製的物件所引用的全部物件都複製一遍。
一. 淺拷貝的實現
淺拷貝的實現方法比較簡單,只要使用是簡單的複製語句即可。
1.1 方法一:簡單的複製語句
/* ================ 淺拷貝 ================ */ 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); // {a: "world", b: 21} console.log(cloneObj.c); // ["Bob", "Tom", "Jenny"] console.log(cloneObj.d); // function() { alert("hello world"); } // 修改拷貝後的物件 cloneObj.b.a = "changed"; cloneObj.c = [1, 2, 3]; cloneObj.d = function() { alert("changed"); }; console.log(obj.b); // {a: "changed", b: 21} // // 原物件所引用的物件被修改了 console.log(obj.c); // ["Bob", "Tom", "Jenny"] // 原物件所引用的物件未被修改 console.log(obj.d); // function() { alert("hello world"); } // 原物件所引用的函式未被修改
1.2 方法二:Object.assign()
Object.assign()
方法可以把任意多個的源物件自身的可列舉屬性拷貝給目標物件,然後返回目標物件。但是 Object.assign()
進行的是淺拷貝,拷貝的是物件的屬性的引用,而不是物件本身。
var obj = { a: {a: "hello", b: 21} }; var initalObj = Object.assign({}, obj); initalObj.a.a = "changed"; console.log(obj.a.a); // "changed"
二. 深拷貝的實現
要實現深拷貝有很多辦法,有最簡單的 JSON.parse()
方法,也有常用的遞迴拷貝方法,和ES5中的 Object.create()
方法。
2.1 方法一:使用 JSON.parse()
方法
要實現深拷貝有很多辦法,比如最簡單的辦法是使用 JSON.parse()
:
/* ================ 深拷貝 ================ */ function deepClone(initalObj) { var obj = {}; try { obj = JSON.parse(JSON.stringify(initalObj)); } return obj; }
/* ================ 客戶端呼叫 ================ */ var obj = { a: { a: "world", b: 21 } } var cloneObj = deepClone(obj); cloneObj.a.a = "changed"; console.log(obj.a.a); // "world"
這種方法簡單易用。
但是這種方法也有不少壞處,譬如它會拋棄物件的constructor。也就是深拷貝之後,不管這個物件原來的建構函式是什麼,在深拷貝之後都會變成Object。
這種方法能正確處理的物件只有 Number, String, Boolean, Array, 扁平物件
,即那些能夠被 json 直接表示的資料結構。RegExp物件是無法通過這種方式深拷貝。
2.2 方法二:遞迴拷貝
程式碼如下:
/* ================ 深拷貝 ================ */ 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; }
上述程式碼確實可以實現深拷貝。但是當遇到兩個互相引用的物件,會出現死迴圈的情況。
為了避免相互引用的物件導致死迴圈的情況,則應該在遍歷的時候判斷是否相互引用物件,如果是則退出迴圈。
改進版程式碼如下:
/* ================ 深拷貝 ================ */ 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; }
2.3 方法三:使用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.extend()方法的實現
jQuery.js的jQuery.extend()也實現了物件的深拷貝。下面將官方程式碼貼出來,以供參考。
官方連結地址:https://github.com/jquery/jquery/blob/master/src/core.js。
jQuery.extend = jQuery.fn.extend = function() { var options, name, src, copy, copyIsArray, clone, target = arguments[ 0 ] || {}, i = 1, length = arguments.length, deep = false; // Handle a deep copy situation if ( typeof target === "boolean" ) { deep = target; // Skip the boolean and the target target = arguments[ i ] || {}; i++; } // Handle case when target is a string or something (possible in deep copy) if ( typeof target !== "object" && !jQuery.isFunction( target ) ) { target = {}; } // Extend jQuery itself if only one argument is passed if ( i === length ) { target = this; i--; } for ( ; i < length; i++ ) { // Only deal with non-null/undefined values if ( ( options = arguments[ i ] ) != null ) { // Extend the base object for ( name in options ) { src = target[ name ]; copy = options[ name ]; // Prevent never-ending loop if ( target === copy ) { continue; } // Recurse if we're merging plain objects or arrays if ( deep && copy && ( jQuery.isPlainObject( copy ) || ( copyIsArray = jQuery.isArray( copy ) ) ) ) { if ( copyIsArray ) { copyIsArray = false; clone = src && jQuery.isArray( src ) ? src : []; } else { clone = src && jQuery.isPlainObject( src ) ? src : {}; } // Never move original objects, clone them target[ name ] = jQuery.extend( deep, clone, copy ); // Don't bring in undefined values } else if ( copy !== undefined ) { target[ name ] = copy; } } } } // Return the modified object return target; };
相關文章
- 【JavaScript】物件的淺拷貝與深拷貝JavaScript物件
- JavaScript物件的深拷貝以及淺拷貝分析JavaScript物件
- JavaScript中的淺拷貝與深拷貝JavaScript
- 物件的深拷貝與淺拷貝物件
- javascript對深拷貝物件的研坑JavaScript物件
- 物件深拷貝和淺拷貝物件
- JavaScript中物件的拷貝JavaScript物件
- Javascript 深拷貝JavaScript
- javascript 淺拷貝VS深拷貝JavaScript
- JavaScript深拷貝和淺拷貝JavaScript
- JavaScript淺拷貝和深拷貝JavaScript
- 實現物件淺拷貝、深拷貝物件
- 聊聊物件深拷貝和淺拷貝物件
- PHP 物件導向 - 物件的淺拷貝與深拷貝PHP物件
- JavaScript之深拷貝和淺拷貝JavaScript
- javascript 生深拷貝JavaScript
- JavaScript深淺拷貝JavaScript
- 關於javascript的深拷貝淺拷貝 思考JavaScript
- JavaScript:利用遞迴實現物件深拷貝JavaScript遞迴物件
- 淺拷貝與深拷貝程式碼(javascript)JavaScript
- jquery之物件拷貝深拷貝淺拷貝案例講解jQuery物件
- 詳解js中物件的深淺拷貝JS物件
- VUE 中 的深拷貝和淺拷貝Vue
- 詳解js中的物件的深淺拷貝JS物件
- 低門檻徹底理解JavaScript中的深拷貝和淺拷貝JavaScript
- 低門檻徹底理解 JavaScript 中的深拷貝和淺拷貝JavaScript
- javaScript深拷貝和淺拷貝簡單梳理JavaScript
- Javascript知識點:淺拷貝和深拷貝JavaScript
- JS中的深拷貝JS
- 理解JS中的淺拷貝與深拷貝JS
- PHP中的淺拷貝和深拷貝薦PHP
- Python - 物件賦值、淺拷貝、深拷貝的區別Python物件賦值
- JavaScript深拷貝的幾種方法JavaScript
- 處理 JavaScript 複雜物件:深拷貝、Immutable & ImmerJavaScript物件
- 淺談Java中的淺拷貝和深拷貝Java
- javascript深淺拷貝講解JavaScript
- 三目運算、物件克隆、深拷貝和淺拷貝物件
- js中的深淺拷貝JS