javascript對(duì)淺拷貝和深拷貝的詳解
下面小編就為大家?guī)?lái)一篇淺談JavaScript中面向?qū)ο蟮牡纳羁截惡蜏\拷貝。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。
1.淺拷貝:復(fù)制一份引用,所有引用對(duì)象都指向一份數(shù)據(jù),并且都可以修改這份數(shù)據(jù)。
2.深拷貝(復(fù)雜):復(fù)制變量值,對(duì)于非基本類型的變量,則遞歸至基本類型變量后,再?gòu)?fù)制。
這里畫一個(gè)簡(jiǎn)單的圖來(lái)加深理解:

一、數(shù)組的深淺拷貝
在使用JavaScript對(duì)數(shù)組進(jìn)行操作的時(shí)候,我們經(jīng)常需要將數(shù)組進(jìn)行備份,事實(shí)證明如果只是簡(jiǎn)單的將它賦予其他變量,那么我們只要更改其中的任何一個(gè),然后其他的也會(huì)跟著改變,這就導(dǎo)致了問題的發(fā)生。
var arr = ["One","Two","Three"];
var arrto = arr;
arrto[1] = "test";
document.writeln("數(shù)組的原始值:" + arr + "<br />");//Export:數(shù)組的原始值:One,test,Three
document.writeln("數(shù)組的新值:" + arrto + "<br />");//Export:數(shù)組的新值:One,test,Three
像上面的這種直接賦值的方式就是淺拷貝,很多時(shí)候,這樣并不是我們想要得到的結(jié)果,其實(shí)我們想要的是arr的值不變,不是嗎?
方法一:js的slice函數(shù)
var arr = ["One","Two","Three"];
var arrtoo = arr.slice(0);
arrtoo[1] = "set Map";
document.writeln("數(shù)組的原始值:" + arr + "<br />");//Export:數(shù)組的原始值:One,Two,Three
document.writeln("數(shù)組的新值:" + arrtoo + "<br />");//Export:數(shù)組的新值:One,set Map,Three
方法二:js的concat方法
var arr = ["One","Two","Three"];
var arrtooo = arr.concat();
arrtooo[1] = "set Map To";
document.writeln("數(shù)組的原始值:" + arr + "<br />");//Export:數(shù)組的原始值:One,Two,Three
document.writeln("數(shù)組的新值:" + arrtooo + "<br />");//Export:數(shù)組的新值:One,set Map To,Three
二、對(duì)象的深淺拷貝
var a={name:'yy',age:26};
var b=new Object();
b.name=a.name;
b.age=a.age;
a.name='xx';
console.log(b);//Object { name="yy", age=26}
console.log(a);//Object { name="xx", age=26}
就是把對(duì)象的屬性遍歷一遍,賦給一個(gè)新的對(duì)象。
var deepCopy= function(source) {
var result={};
for (var key in source) {
result[key] = typeof source[key]==='object'? deepCoyp(source[key]): source[key];
}
return result;
}
舉一個(gè)jQuery中的例子:
jQuery.extend = jQuery.fn.extend = function() {//1.將extend方法擴(kuò)展到JQ(函數(shù))下邊:擴(kuò)展靜態(tài)方法
//2. jQuery.fn.extend 把extend擴(kuò)展到j(luò)q.fn下 且jQuery.fn = jQuery.prototype 擴(kuò)展實(shí)例方法
// 1.2.功能相似
var options, name, src, copy, copyIsArray, clone, //定義一些變量
target = arguments[0] || {},
//目標(biāo)元素是【0】第一個(gè)元素$.extend( a , { name : 'hello' } , { age : 30 } );
i = 1, //第一個(gè)元素的位置
length = arguments.length,//第一個(gè)個(gè)對(duì)象的元素
deep = false; //是否是深拷貝 默認(rèn) false不是
// Handle a deep copy situation 看是不是深拷貝情況
if ( typeof target === "boolean" ) { //是布爾值
deep = target;
target = arguments[1] || {}; //目標(biāo)元素是第二個(gè)$.extend( true , a , b )
// skip the boolean and the target
i = 2;
}
// Handle case when target is a string or something (possible in deep copy) 看參數(shù)正確不
if ( typeof target !== "object" && !jQuery.isFunction(target) ) {
// 當(dāng)目標(biāo)不是對(duì)象或者不是函數(shù)的時(shí)候
target = {}; //變成一個(gè)空的jason
}
// extend jQuery itself if only one argument is passed看是不是插件情況
if ( length === i ) { //只寫了一個(gè)對(duì)象 要把這個(gè)對(duì)象擴(kuò)展到j(luò)q源碼上 靜態(tài)方法 或者是實(shí)例方法
target = this; //this 是$ 或者 $();
--i;
}
// 可能有多個(gè)對(duì)象情況
for ( ; i < length; i++ ) {
// Only deal with non-null/undefined values
if ( (options = arguments[ i ]) != null ) {//看后邊的對(duì)象是否都有值
// Extend the base object
for ( name in options ) {
src = target[ name ];
copy = options[ name ];
// Prevent never-ending loop
if ( target === copy ) {//防止循環(huán)引用
continue;//跳出本次循環(huán)繼續(xù)執(zhí)行
// $.extend( a , { name : a } ) );循環(huán)引用 a也是一個(gè)對(duì)象
}
// Recurse if we're merging plain objects or arrays深拷貝
if ( deep && copy && ( jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)) ) ) {
// 是深拷貝 且需有var b = { name : { age : 30 } }; 且b必須是對(duì)象自變量(jason) 或者是個(gè)數(shù)組
//遞歸
if ( copyIsArray ) { //數(shù)組
copyIsArray = false;
clone = src && jQuery.isArray(src) ? src : []; //定義一個(gè)空數(shù)組
} else {//jason
clone = src && jQuery.isPlainObject(src) ? src : {};//看原有的屬性有沒有且是不是jason定義一個(gè)空jason
}
// var a = { name : { job : 'it' } }; 看有沒有原有的屬性 有的話在原有的上邊添加
// var b = { name : {age : 30} };
// $.extend( true , a , b );//a繼承b
// console.log( a ); a{ name:{ job : 'it' ,age : 30}} 如果只有一個(gè){} 則只有,age : 30
// Never move original objects, clone(a) them
target[ name ] = jQuery.extend( deep, clone, copy );
//調(diào)用函數(shù)本身進(jìn)行進(jìn)一步的遞歸處理
// Don't bring in undefined values淺拷貝
} else if ( copy !== undefined ) {
target[ name ] = copy; //直接復(fù)制因?yàn)槔镞厸]有對(duì)象
}
}
}
}
// Return the modified object
return target;
};
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Javascript 淺拷貝、深拷貝的實(shí)現(xiàn)代碼
- JavaScript數(shù)組深拷貝和淺拷貝的兩種方法
- 淺談JavaScript中面向?qū)ο蟮牡纳羁截惡蜏\拷貝
- js對(duì)象淺拷貝和深拷貝詳解
- javascript深拷貝和淺拷貝詳解
- 淺析javaScript中的淺拷貝和深拷貝
- JavaScript基礎(chǔ)心法 深淺拷貝(淺拷貝和深拷貝)
- javascript深拷貝、淺拷貝和循環(huán)引用深入理解
- JavaScript深拷貝和淺拷貝概念與用法實(shí)例分析
- JavaScript實(shí)現(xiàn)淺拷貝與深拷貝的方法分析
- JS淺拷貝和深拷貝原理與實(shí)現(xiàn)方法分析
- JS中實(shí)現(xiàn)淺拷貝和深拷貝的代碼詳解
- javascript二維數(shù)組和對(duì)象的深拷貝與淺拷貝實(shí)例分析
- javascript 關(guān)于賦值、淺拷貝、深拷貝的個(gè)人理解
- JS賦值、淺拷貝和深拷貝(數(shù)組和對(duì)象的深淺拷貝)實(shí)例詳解
- 詳解JS深拷貝與淺拷貝
- JS對(duì)象復(fù)制(深拷貝和淺拷貝)
- js深拷貝與淺拷貝一文徹底搞懂
相關(guān)文章
javascript:void(0)的含義及用法實(shí)例
這篇文章介紹了javascript:void(0)的含義及用法實(shí)例,對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-12-12
window.requestAnimationFrame是什么意思,怎么用
window.requestAnimationFrame 告訴瀏覽器您要執(zhí)行的動(dòng)畫并且請(qǐng)求瀏覽器的在下一個(gè)動(dòng)畫幀重繪窗口,方法在瀏覽器重繪之前作為一個(gè)回調(diào)函數(shù)被調(diào)用,就是告訴瀏覽器在刷新屏幕的時(shí)候,調(diào)用這個(gè)方法2013-01-01
網(wǎng)頁(yè)中表單按回車就自動(dòng)提交的問題的解決方案
這篇文章主要介紹了網(wǎng)頁(yè)中表單按回車就自動(dòng)提交的問題的解決方案,需要的朋友可以參考下2014-11-11
javascript中對(duì)Date類型的常用操作小結(jié)
下面小編就為大家?guī)?lái)一篇javascript中對(duì)Date類型的常用操作小結(jié)。小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來(lái)看看吧2016-05-05

