亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

詳解JavaScript對象的深淺復制

發布時間:2020-08-20 18:10:06 來源:腳本之家 閱讀:144 作者:Claiyre 欄目:web開發

前言

從層次上來看,對象的復制可以簡單地分為淺復制和深復制,顧名思義,淺復制是指只復制一層對象的屬性,不會復制對象中的對象的屬性,對象的深復制會復制對象中層層嵌套的對象的屬性。

在復制對象時,除了要復制對象的屬性外,還要兼顧到是否保留了對象的constructor屬性,是否對每一種數據類型(JavaScript常見的數據類型有String,Number,Boolean,Data,RegExp,Array,Funtion,Object)都實現正確的復制。項目中,我們可以根據實際情況,決定需要實現什么樣程度的復制。

本文是我在復制對象方面的一些心得總結,由淺復制到深復制,由只復制簡單屬性到復制Function,RegExp等復雜屬性,層層遞進。如有陳述不當之處,煩請指出,不勝感激。

正文

淺復制

淺復制只會依次復制對象的每一個屬性,不會對這些屬性進行遞歸復制。下面是一個簡單的淺復制實現。

//對象淺復制        
function shadowCopy(obj){
 if(typeof obj !== 'object') return obj;
 for(var prop in obj){
  if(obj.hasOwnProperty(prop)){
  newObj[prop] = obj[prop];
  }
 }
 return newObj;
 }

仔細觀察,不難發現上述方法的缺陷:

1.不能正確實現數組的淺復制

2.復制操作丟失了對象的constructor屬性

好,我們現在已經發現了問題所在,只需針對性地解決,一個還算完美的淺復制對象的方法就誕生了!

//對象淺復制
 function shadowCopy(obj){
  if(typeof obj !== 'object') return ;
  var newObj;
  //保留對象的constructor屬性
  if(obj.constructor === Array){
  newObj = [];
  } else {
  newObj = {};
  newObj.constructor = obj.constructor;
  }
  for(var prop in obj){
  if(obj.hasOwnProperty(prop)){
   newObj[prop] = obj[prop];
  }
  }
  return newObj;
 }

瀏覽器中測試一下:

var arr1 = [0,1,2];
 console.log(arr1);
 console.log(shadowCopy(arr1));
 var arr2 = [0,1,2,[3,4,5]],
 arr2Copy = shadowCopy(arr2);
 console.log(arr2);
 console.log(arr2Copy);
 arr2Copy[3][0] = 6;
 console.log(arr2[3][0]); //6

詳解JavaScript對象的深淺復制

Good! 可以正確實現數組復制和并且保留constructor了,但細心的你一定發現了,淺復制后的對象的 arr2Copy[3] 和 arr2[3] 指向的是一個對象,改變其中一個,同時也會改變另一個。我們想要實現的是 復制,但這并不是復制呀!
這是淺復制的一個弊端所在,接下讓我們看看深復制是怎樣解決這個問題的。

深復制

深復制需要層層遞歸,復制對象的所有屬性,包括對象屬性的屬性的屬性....(暈~)
如果只是需要簡單地復制對象的屬性,而不用考慮它的constructor,也不用考慮函數,正則,Data等特殊數據類型,那這里有一個深復制的小trick,兩行代碼即可:

function deepCopy(obj){
 if(typeof obj !== "object"){ return ;}
 var str = JSON.stringify(obj);
 return JSON.parse(str);
}

大多數情況下,上面的就可以滿足要求了,但一些時候,我們需要把函數,正則等特殊數據類型也考慮在內,或者當前環境不支持JSON時,上面的方法也就不適用了。這時,我們可以通過遞歸來實現對象的深層復制,如下:

function deepCopy(obj){
 if(typeof obj !== "object"){ return ;}
 var newObj;
 //保留對象的constructor屬性
 if(obj.constructor === Array){
 newObj = [];
 } else {
 newObj = {};
 newObj.constructor = obj.constructor;
 }
 for(var prop in obj){
 if(typeof obj[prop] === 'object'){
  if(obj[prop].constructor === RegExp ||obj[prop].constructor === Date){
  newObj[prop] = obj[prop];
  } else {
  //遞歸
  newObj[prop] = deepCopy(obj[prop]);
  }
 } else {
  newObj[prop] = obj[prop];
 }
 }
 return newObj;
}

先用上面的例子測試:

詳解JavaScript對象的深淺復制

棒!可以正確實現多維數組的復制,再看是否能實現函數和正則的復制:

function Person(name){
 this.name = name;
 this.age = age;
 this.search = new RegExp(name);
 this.say = function(){
 console.log(this.name + "今年" + this.age + "歲了");
 }
}
var p1 = new Person("Claiyre",20),
 p2 = deepCopy(p1);
console.log(p1);
console.log(p2);
p2.age = 22;
p1.say();
p2.say();

詳解JavaScript對象的深淺復制

圓滿完成!!

稍加整理,我們就可以得到一個較為通用的js對象復制函數:

function deepCopy(obj){
 var newObj = obj.constructor === Array ? []:{};
 newObj.constructor = obj.constructor;
 if(typeof obj !== "object"){ 
 return ;
 } else if(window.JSON){
 //若需要考慮特殊的數據類型,如正則,函數等,需把這個else if去掉即可
 newObj = JSON.parse(JSON.stringify(obj));
 } else {
 for(var prop in obj){
  if(obj[prop].constructor === RegExp ||obj[prop].constructor === Date){
  newObj[prop] = obj[prop];
  } else if(typeof obj[prop] === 'object'){
  //遞歸
  newObj[prop] = deepCopy(obj[prop]);
  } else {
  newObj[prop] = obj[prop];
  }
 }
 } 
 return newObj;
}

結語

面向對象的編程語言,其核心是對象,因此深入了解對象的相關操作,縱向比較異同,對學習過程是極有好處的。

以上所述是小編給大家介紹的JavaScript對象的深淺復制,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

枣庄市| 侯马市| 江川县| 孟村| 景德镇市| 北流市| 洞头县| 侯马市| 娄烦县| 平遥县| 寿阳县| 阿拉善左旗| 民权县| 苏尼特左旗| 阳信县| 同江市| 正宁县| 金阳县| 长治市| 临夏市| 桦南县| 渭南市| 广昌县| 肃北| 吉林省| 玉门市| 河间市| 无棣县| 吉木萨尔县| 屯留县| 红河县| 和政县| 长白| 阳城县| 丰县| 宁都县| 上杭县| 合作市| 丘北县| 竹山县| 克什克腾旗|