JavaScript 復制對象與Object.assign方法無法實現深復制

    在JavaScript這門語言中,數據類型分為兩大類:基本數據類型和復雜數據類型。愛掏網 - it200.com基本數據類型包括Number、Boolean、String、Null、String、Symbol(ES6 新增),而復雜數據類型包括Object,而所有其他引用類型(Array、Date、RegExp、Function、基本包裝類型(Boolean、String、Number)、Math等)都是Object類型的實例對象,因此都可以繼承Object原型對象的一些屬性和方法。愛掏網 - it200.com

    而對于基本數據類型來說,復制一個變量值,本質上就是copy了這個變量。愛掏網 - it200.com一個變量值的修改,不會影響到另!
    外一個變量。愛掏網 - it200.com看一個簡單的例子。愛掏網 - it200.com

    let val = 123;
    let copy = val;
    console.log(copy); //123
    val = 456;     //修改val的值對copy的值不產生影響
    console.log(copy); //123

    而對于復雜數據類型來說,同基本數據類型實現的不太相同。愛掏網 - it200.com對于復雜數據類型的復制,要注意的是,變量名只是指向這個對象的指針。愛掏網 - it200.com當我們將保存對象的一個變量賦值給另一個變量時,實際上復制的是這個指針,而兩個變量都指向都一個對象。愛掏網 - it200.com因此,一個對象的修改,會影響到另外一個對象。愛掏網 - it200.com

    // obj只是指向對象的指針
    let obj = {
      character: 'peaceful'
    };
    //copy變量復制了這個指針,指向同一個對象
    let copy = obj;
    console.log(copy);     //{character: 'peaceful'}
    obj.character = 'lovely';
    console.log(copy);     //{character: 'lovely'}

    有一副很形象的圖描述了復雜數據類型復制的原理
    同理,在復制一個數組時,變量名只是指向這個數組對象的指針;在復制一個函數時,函數名只是指向這個函數對象的指針

    let arr = [1, 2, 3];
    let copy = arr;
    console.log(copy); // [1, 2, 3]
    arr[0] = 'keith';
    console.log(copy); // 數組對象被改變: ['keith', 2, 3]
    arr = null;
    歡迎加入全棧開發交流劃水交流圈:582735936
    面向劃水1-3年前端人員
    幫助突破劃水瓶頸,提升思維能力
    console.log(copy); // ['keith', 2, 3] 即使arr=null,也不會影響copy。愛掏網 - it200.com因此此時的arr變量只是一個指向數組對象的指針
    
    function foo () {
      return 'hello world';
    };
    let bar = foo;
    console.log(foo());
    foo = null;   //foo只是指向函數對象的指針
    console.log(bar());

    因此,我們應該如何實現對象的深淺復制?

    復制對象

    在JavaScript中,復制對象分為兩種方式,淺復制和深復制。愛掏網 - it200.com

    淺復制沒有辦法去真正的去復制一個對象,而只是保存了對該對象的引用;而深復制可以實現真正的復制一個對象。愛掏網 - it200.com

    淺復制

    在ES6中,Object對象新增了一個assign方法,可以實現對象的淺復制。愛掏網 - it200.com這里談談Object.assign方法的具體用法,因為稍后會分析jQuery的extend方法,實現的原理同Object.assign方法差不多

    Object.assign的第一個參數是目標對象,可以跟一或多個源對象作為參數,將源對象的所有可枚舉([[emuerable]] === true)復制到目標對象。愛掏網 - it200.com這種復制屬于淺復制,復制對象時只是包含對該對象的引用。愛掏網 - it200.comObject.assign(target, [source1, source2, ...])

    如果目標對象與源對象有同名屬性,則后面的屬性會覆蓋前面的屬性
    如果只有一個參數,則直接返回該參數。愛掏網 - it200.com即Object.assign(obj) === obj
    如果第一個參數不是對象,而是基本數據類型(Null、Undefined除外),則會調用對應的基本包裝類型
    如果第一個參數是Null和Undefined,則會報錯;如果Null和Undefined不是位于第一個參數,則會略過該參數的復制
    要實現對象的淺復制,可以使用Object.assign方法

    let target = {a: 123};
    let source1 = {b: 456};
    let source2 = {c: 789};
    歡迎加入全棧開發交流劃水交流圈:582735936
    面向劃水1-3年前端人員
    幫助突破劃水瓶頸,提升思維能力
    let obj = Object.assign(target, source1, source2);
    console.log(obj);

    不過對于深復制來說,Object.assign方法無法實現

    let target = {a: 123};
    let source1 = {b: 456};
    let source2 = {c: 789, d: {e: 'lovely'}};
    let obj = Object.assign(target, source1, source2);
    source2.d.e = 'peaceful';
    console.log(obj);  // {a: 123, b: 456, c: 789, d: {e: 'peaceful'}}

    從上面代碼中可以看出,source2對象中e屬性的改變,仍然會影響到obj對象

    深復制

    在實際的開發項目中,前后端進行數據傳輸,主要是通過JSON實現的。愛掏網 - it200.comJSON全稱:JavaScript Object Notation,JavaScript對象表示法。愛掏網 - it200.com

    JSON對象下有兩個方法,一是將JS對象轉換成字符串對象的JSON.stringify方法;一個是將字符串對象轉換成JS對象的JSON.parse方法。愛掏網 - it200.com

    這兩個方法結合使用可以實現對象的深復制。愛掏網 - it200.com也就是說,當我們需要復制一個obj對象時,可以先調用JSON.stringify(obj),將其轉換為字符串對象,然后再調用JSON.parse方法,將其轉換為JS對象。愛掏網 - it200.com就可以輕松的實現對象的深復制

    let obj = {
      a: 123,
      b: {
        c: 456,
        d: {
          e: 789
        }
      }
    };
    let copy = JSON.parse(JSON.stringify(obj));
    // 對obj對象無論怎么修改,都不會影響到copy對象
    obj.b.c = 'hello';
    obj.b.d.e = 'world';
    console.log(copy); // {a: 123, b: {c: 456, d: {e: 789}}}

    當然,使用這種方式實現深復制有一個缺點就是必須給JSON.parse方法傳入的字符串必須是合法的JSON,否則會拋出錯誤

    jQuery.extend || jQuery.fn.extend

    jQuery.extend對象,對使用jQuery超過一定時間的朋友來說并不默認。愛掏網 - it200.com這個$.extend方法可以用來擴展jQuery的全局對象,而$.fn.extend方法可以用來擴展實例對象。愛掏網 - it200.comfn實際上是prototype對象的別名,所以,擴展實例對象的方法實際上就是在jQuery原型對象上添加一些方法。愛掏網 - it200.com

    $.extend方法不僅可以用來寫jQuery插件,同樣的,它可以用來實現對象的深淺復制。愛掏網 - it200.com(使用$.extend與$.fn.extend實現深淺復制都可以,唯一的差別就是this的指向性不同)

    在具體分析源代碼之前,我在源碼中看到的$.extend方法的一些特點

    當不接受任何參數時,直接返回一個空對象
    當只有一個參數時(這個參數可以任何數據類型(Null、Undefined、Boolean、String、Number、Object)),會返回this對象,這里會分為兩種情況。愛掏網 - it200.com如果用$.extend,會返回jQuery對象;如果用$.fn.extend,會返回jQuery的原型對象。愛掏網 - it200.com
    當接收兩個參數時,并且第一個參數是Boolean值時,也會返回一個空對象。愛掏網 - it200.com如果第一個參數不是Boolean值,那么會將源對象復制到目標對象
    當接收三個參數以上時,可以分為兩種情況。愛掏網 - it200.com如果第一個參數是Boolean值表示深淺復制,那么目標對象會移動到第二個參數,源對象會移動到第三個參數。愛掏網 - it200.com(目標對象、源對象和Object.assign方法中的相同)。愛掏網 - it200.com如果第一個參數不是Boolean值,那么用法與Object.assign方法常規的復制相同。愛掏網 - it200.com
    在循環源對象的過程中,任何數據類型為Null、Undefined或者源對象是一個空對象時,在復制的過程中都會被忽略。愛掏網 - it200.com
    如果源對象和目標對象具有同名的屬性,則源對象的屬性會覆蓋掉目標對象中的屬性。愛掏網 - it200.com如果同名屬性是一個對象的話,則會在deep=true等其他條件下向目標對象的該同名對象添加屬性
    下面貼出jQuery-2.1.4中jQuery.extend實現方式的源代碼

    jQuery.extend = jQuery.fn.extend = function() {
      var options, name, src, copy, copyIsArray, clone,
        target = arguments[0] || {},
        // 使用||運算符,排除隱式強制類型轉換為false的數據類型
        // 如'', 0, undefined, null, false等
        // 如果target為以上的值,則設置target = {}
        i = 1,
        length = arguments.length,
        deep = false;
    
      // 當typeof target === 'boolean'時
      // 則將deep設置為target的值
      // 然后將target移動到第二個參數,
      if (typeof target === "boolean") {
        deep = target;
        // 使用||運算符,排除隱式強制類型轉換為false的數據類型
        // 如'', 0, undefined, null, false等
        // 如果target為以上的值,則設置target = {}
        target = arguments[i] || {};
        i++;
      }
    
      // 如果target不是一個對象或數組或函數,
      // 則設置target = {}
      // 這里與Object.assign的處理方法不同,
      // assign方法會將Boolean、String、Number方法轉換為對應的基本包裝類型
      // 然后再返回,
      // 而extend方法直接將typeof不為object或function的數據類型
      // 全部轉換為一個空對象
      if (typeof target !== "object" && !jQuery.isFunction(target)) {
        target = {};
      }
    
      // 如果arguments.length === 1 或
      // typeof arguments[0] === 'boolean', 且存在arguments[1],
      // 這時候目標對象會指向this
      // this的指向哪個對象需要看是使用$.fn.extend還是$.extend
      if (i === length) {
        target = this;
        // i-- 表示不進入for循環
        i--;
      }
    
      // 循環arguments類數組對象,從源對象開始
      for (; i 
    

    因此,可以針對分析過后的源碼,給出一些例子

    let obj1 = $.extend();
    console.log(obj1); // 返回一個空對象 {}
    
    let obj2 = $.extend(undefined);
    console.log(obj2); //返回jQuery對象,Object.assign傳入undefined會報錯
    
    let obj3 = $.extend('123');
    console.log(obj3); // 返回jQuery對象,Object.assign傳入'123'會返回字符串的String對象
    
    let target = {
      a: 123,
      b: 234
    };
    
    let source1 = {
      b: 456,
      d: ['keith', 'peaceful', 'lovely']
    };
    
    let source2 = {c: 789};
    let source3 = {};
    
    let obj4 = $.extend(target, source1, source2);
    // let obj4 = $.extend(false, target, source1, source2);
    console.log(obj4); // {a: 123, b: 456, d: Array(3), c: 789}
    // 默認情況下,復制方式都是淺復制
    // 如果只需要淺復制,不傳入deep參數也可以
    // 淺復制時,obj4對象中的d屬性只是指向數組對象的指針
    
    let obj5 = $.extend(target, undefined, source2);
    let obj6 = $.extend(target, source3, source2);
    console.log(obj5, obj6);
    // {a: 123, b: 234, c: 789}, {a: 123, b: 234, c: 789}
    // 會略過空對象或Undefined、Null值
    
    let obj7 = $.extend(true, target, source1, source2);
    console.log(obj7); // {a: 123, b: 456, d: Array(3), c: 789}
    // 這里target對象有b屬性,源對象source1也有b屬性
    // 此時源對象的b屬性會覆蓋目標對象的b屬性
    // 這里deep=true,屬于深復制
    // 當name=d時,會遞歸調用$.extend, 直到它的屬性對應的屬性值全部為基本數據類型
    // 源對象的改變不會影響到obj7對象

    JavaScript 復制對象

    因此,可以根據$.extend方法,寫出一個通用的實現對象深淺復制的函數,copyObject函數唯一的不同就是當i === arguments.length屬性時,copyObject函數直接返回了target對象

    function copyObject () {
      let i = 1,
        target = arguments[0] || {},
        deep = false,
        length = arguments.length,
        name, options, src, copy,
        copyIsArray, clone;
    
      // 如果第一個參數的數據類型是Boolean類型
      // target往后取第二個參數
      if (typeof target === 'boolean') {
        deep = target;
        // 使用||運算符,排除隱式強制類型轉換為false的數據類型
        // 如'', 0, undefined, null, false等
        // 如果target為以上的值,則設置target = {}
        target = arguments[1] || {};
        i++;
      }
    
      // 如果target不是一個對象或數組或函數
      if (typeof target !== 'object' && !(typeof target === 'function')) {
        target = {};
      }
    
      // 如果arguments.length === 1 或
      // typeof arguments[0] === 'boolean',
      // 且存在arguments[1],則直接返回target對象
      if (i === length) {
        return target;
      }
    
      // 循環每個源對象
      for (; i 
    

    以上就是本文的全部內容,希望對大家的學習有所幫助

    聲明:所有內容來自互聯網搜索結果,不保證100%準確性,僅供參考。如若本站內容侵犯了原著者的合法權益,可聯系我們進行處理。
    發表評論
    更多 網友評論0 條評論)
    暫無評論

    返回頂部

    主站蜘蛛池模板: 免费人妻精品一区二区三区| 97av麻豆蜜桃一区二区| 亚洲色欲一区二区三区在线观看| 中文字幕在线一区二区在线 | 日本一区二区三区不卡视频中文字幕 | 国产99精品一区二区三区免费| 国产精品女同一区二区| 亚洲一区无码精品色| 美女视频一区二区三区| 国产精品福利一区| 爱爱帝国亚洲一区二区三区| 欧亚精品一区三区免费| 日本无卡码一区二区三区| 亚洲AV无码一区二区三区DV| 嫩B人妻精品一区二区三区| 日韩精品一区二区三区视频| 国产综合精品一区二区| 国产成人av一区二区三区在线观看| 国产精品合集一区二区三区| 色国产精品一区在线观看| 日产精品久久久一区二区| 亚洲av一综合av一区| 蜜桃臀无码内射一区二区三区| 精品一区二区三区四区| 无码av免费一区二区三区试看| 国产成人精品无码一区二区三区 | 亚洲国产福利精品一区二区| 精品一区二区三区波多野结衣| 91视频一区二区| 国产成人综合一区精品| 国产精品无码一区二区三区不卡 | 日韩美一区二区三区| 国产一区二区影院| 一区二区三区在线观看视频| 中文字幕一区二区三区四区| 亚洲AV综合色区无码一区爱AV| 亚洲中文字幕在线无码一区二区| 国产乱人伦精品一区二区 | 美女毛片一区二区三区四区| 麻豆一区二区免费播放网站| 国产在线一区二区视频|