js-引用類型對象拷貝

1.引用類型; 2.過濾數(shù)組; 3.深淺拷貝方法;

1.引用類型有哪些?非引用類型有哪些

  • 引用類型:
    引用類型(Object, Array, Function, Date, Math, RegExp, Error)指的是那些保存在堆內(nèi)存中的對象,變量中保存的實際上只是一個指針,這個指針指向內(nèi)存中的另一個位置,由該位置保存對象。

  • 非引用類型(基本數(shù)據(jù)類型):
    number,string,boolean,undefined, null.
    指的是保存在棧內(nèi)存中的簡單數(shù)據(jù)段.

  • 基本類型 == 非引用類型
    復(fù)雜類型 == 對象 == 引用類型

2.如下代碼輸出什么?為什么

var obj1 = {a:1, b:2};
var obj2 = {a:1, b:2};
console.log(obj1 == obj2);//輸出:fasle:會比較兩者內(nèi)存地址,發(fā)現(xiàn)內(nèi)存地址不一樣.

console.log(obj1 = obj2);// 輸出 : Object {a: 1,b: 2}
:這一步把obj2的內(nèi)存地址給了obj1,從此兩者共用一份內(nèi)存地址.

console.log(obj1 == obj2);//輸出:ture :因為兩者共用一份內(nèi)存地址了

3.如下代碼輸出什么? 為什么

var a = 1;
var b = 2;
var c = { name: '饑人谷', age: 2 };
var d = [a, b, c];

var aa = a;//a=1
var bb = b;//a=2
var cc = c;
var dd = d;

a = 11;//aa=1:基本類型引用,只會讓a=11,aa和a 是指向兩塊不同的內(nèi)存空間(即指針).不干擾
b = 22;//bb=2
c.name = 'hello';//{name: "hello", age: 2}
d[2]['age'] = 3;//這句把c對象字面量改為{name: "hello", age: 3}

console.log(aa); //1 
console.log(bb);//2
console.log(cc);//Object{name: "hello", age: 3}  :引用類型中:兩個變量指向同一個堆對象,改變其中一個變量,另一個也會受到影響
console.log(dd);//[1,2,Object{age: 3, name: "hello"}]



4.如下代碼輸出什么? 為什么

var a = 1;
var c = { name: 'jirengu', age: 2 };

function f1(n){
  //var n =a;相當于有這步,但并沒有給a開辟新的內(nèi)存.
  ++n;
}
function f2(obj){
  ++obj.age;
}

f1(a);//a=1  n=2
f2(c);//c={name: 'jirengu', age: 2 };
f1(c.age);//c={name: 'jirengu', age: 3 };對象a的點語法訪問的是同一個內(nèi)存地址,所以原指針c的age屬性會改變.
console.log(a);//a=1
console.log(c);//Object {name: 'jirengu', age: 3 };

5.過濾如下數(shù)組,只保留正數(shù),直接在原數(shù)組上操作

 var arr = [3,1,0,-1,-3,2,-5];
  
  function filter(arr){

    for(var i=0;i<arr.length;i++){
      if( arr[i]<1 ) {
        arr.splice(i,1); //splice(i,1)會從第i個開始,刪除1個數(shù),用于數(shù)組增刪元素
        --i;  //當刪除一個元素,arr的長度會減1,但for循環(huán)的i會一直加1,這樣每次刪一個,會漏掉被刪元素后一個的元素的判斷
      }
 
    }

  }
  filter(arr);
  console.log(arr); // [3,1,2]

6.過濾如下數(shù)組,只保留正數(shù),原數(shù)組不變,生成新數(shù)組

方法1:

 var arr = [3,1,0,-1,-3,2,-5];
    function filter(arr){
      var newarr = [];
      for(var i = 0;i<arr.length;i++ ){
         if(arr[i]>0 ){
           newarr.splice(i,0,arr[i]);//splice第二個參數(shù)如果設(shè)置為 0,則不會刪除項目,第三個參數(shù)是添加的參數(shù).
         }
      }
      return newarr;
    }
    var arr2 = filter(arr);
    console.log(arr2); // [3,1,2]
    console.log(arr);  // [3,1,0,-1,-2,2,-5]

方法2:

var arr = [3,1,0,-1,-3,2,-5];
function filter(arr){
    var newArr = [];
    for(var i = 0; i<arr.length;i++){
        if(arr[i]>0){
            newArr.push(arr[i]);
        }
    }
    return newArr;
}
var arr1 = filter(arr);
console.log(arr1);
console.log(arr);

方法3:

var arr = [3,1,0,-1,-3,2,-5];
function filter(arr){
  var j = 0;
  var newarr = [];
  for( var i = 0; i < arr.length; i++){
    if( arr[i] > 0){
      newarr[j] = arr[i];
      j++;
    }
  }
  return newarr;
}
var arr2 = filter(arr);
console.log(arr2); // [3,1,2]
console.log(arr);  // [3,1,0,-1,-2,2,-5]

7.寫一個深拷貝函數(shù),用兩種方式實現(xiàn)

  • 在JavaScript中,對于Object和Array這類引用類型值,當從一個變量向另一個變量復(fù)制引用類型值時,這個值的副本其實是一個指針,兩個變量指向同一個堆對象,改變其中一個變量,另一個也會受到影響

  • 而引用類型拷貝分為兩種
    淺拷貝:拷貝原對象的引用.
    深拷貝:是拷貝出一個新的實例,新的實例和之前的實例互不影響.兩份不同的內(nèi)存地址.

  • 方法一

function deepCope(obj){
    var newObj = {}; 
    for(var key in obj){
     //只賦值對象自己的屬性,原型上的屬性不賦值
      if(obj.hasOwnProperty(key)){//obj.hasOwnProperty(prop)這個方法可以用來檢測一個對象是否含有特定的自身屬性
        if(typeof obj[key] ==== 'number' ||
           typeof obj[key] ==== 'string' ||
           typeof obj[key] ==== 'undefined' ||
           typeof obj[key] ==== 'boolean' ||
                  obj[key] ==== null){//由于typeof null返回值也是object,所以不能直接用typeof obj[key] === 'object'來判斷
         newObj[key] = obj[key];
        }else{
          newObj[key] = deepCope(obj[key]);//遞歸
      }
        
    }
      return newObj;
      
  }



  • 上述的hasOwnProperty 方法判斷屬性是否存在,下面的例子檢測了對象 o 是否含有自身屬性 prop
o = new Object();
o.prop = 'exists';

function changeO() {
  o.newprop = o.prop;
  delete o.prop;
}

o.hasOwnProperty('prop');   // 返回 true
changeO();
o.hasOwnProperty('prop');   // 返回 false
  • 方法二
下面兩個方法可以實現(xiàn)對象的深復(fù)制.
JOSN對象中的stringify可以把一個js對象序列化為一個JSON字符串,parse可以把JSON字符串反序列化為一個js對象
// 利用JSON序列化實現(xiàn)一個深拷貝
function deepClone(source){
  return JSON.parse(JSON.stringify(source));
}
var o1 = {
  arr: [1, 2, 3],
  obj: {
    key: 'value'
  },
  func: function(){
    return 1;
  }
};
var o2 = deepClone(o1);
console.log(o2); // => {arr: [1,2,3], obj: {key: 'value'}}

參考

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,606評論 6 533
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,996評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,991評論 2 374

推薦閱讀更多精彩內(nèi)容

  • 引用類型 引用類型變量保存的僅僅是一個指針,指針指向堆內(nèi)存中保存對象的位置。 所以基本類型復(fù)制的時候僅僅復(fù)制值,復(fù)...
    DeeJay_Y閱讀 584評論 0 0
  • 1.引用類型有哪些?非引用類型有哪些2.如下代碼輸出什么?為什么 引用類型有:對象, 數(shù)組, 函數(shù), 正則引用類型...
    billa_8f6b閱讀 310評論 0 0
  • 1. 引用類型有哪些?非引用類型有哪些 基本類型值(數(shù)值、布爾值、null和undefined):指的是保存在棧內(nèi)...
    蕭雪圣閱讀 472評論 0 1
  • 1.引用類型有哪些?非引用類型有哪些 引用類型是指那些保存在堆內(nèi)存中的對象。變量中保存的實際上只是一個指針,這個指...
    高進哥哥閱讀 317評論 0 0
  • 受臺風(fēng)“妮妲”的影響,今天身處深圳的我們,得以破天荒的不用上班。 不用上班又不能出門的這一天,要怎么...
    懶懶散散的海蠣閱讀 676評論 0 0