溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

JavaScript 復(fù)制對(duì)象與Object.assign方法無(wú)法實(shí)現(xiàn)深復(fù)制

發(fā)布時(shí)間:2020-10-04 20:50:39 來(lái)源:腳本之家 閱讀:195 作者:最騷的就是你 欄目:web開發(fā)

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

而對(duì)于基本數(shù)據(jù)類型來(lái)說(shuō),復(fù)制一個(gè)變量值,本質(zhì)上就是copy了這個(gè)變量。一個(gè)變量值的修改,不會(huì)影響到另外一個(gè)變量??匆粋€(gè)簡(jiǎn)單的例子。

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

而對(duì)于復(fù)雜數(shù)據(jù)類型來(lái)說(shuō),同基本數(shù)據(jù)類型實(shí)現(xiàn)的不太相同。對(duì)于復(fù)雜數(shù)據(jù)類型的復(fù)制,要注意的是,變量名只是指向這個(gè)對(duì)象的指針。當(dāng)我們將保存對(duì)象的一個(gè)變量賦值給另一個(gè)變量時(shí),實(shí)際上復(fù)制的是這個(gè)指針,而兩個(gè)變量都指向都一個(gè)對(duì)象。因此,一個(gè)對(duì)象的修改,會(huì)影響到另外一個(gè)對(duì)象。

// obj只是指向?qū)ο蟮闹羔?let obj = {
  character: 'peaceful'
};
//copy變量復(fù)制了這個(gè)指針,指向同一個(gè)對(duì)象
let copy = obj;
console.log(copy);     //{character: 'peaceful'}
obj.character = 'lovely';
console.log(copy);     //{character: 'lovely'} 

有一副很形象的圖描述了復(fù)雜數(shù)據(jù)類型復(fù)制的原理

JavaScript 復(fù)制對(duì)象與Object.assign方法無(wú)法實(shí)現(xiàn)深復(fù)制

同理,在復(fù)制一個(gè)數(shù)組時(shí),變量名只是指向這個(gè)數(shù)組對(duì)象的指針;在復(fù)制一個(gè)函數(shù)時(shí),函數(shù)名只是指向這個(gè)函數(shù)對(duì)象的指針

let arr = [1, 2, 3];
let copy = arr;
console.log(copy); // [1, 2, 3]
arr[0] = 'keith';
console.log(copy); // 數(shù)組對(duì)象被改變: ['keith', 2, 3]
arr = null;
console.log(copy); // ['keith', 2, 3] 即使arr=null,也不會(huì)影響copy。因此此時(shí)的arr變量只是一個(gè)指向數(shù)組對(duì)象的指針

function foo () {
  return 'hello world';
};
let bar = foo;
console.log(foo());
foo = null;   //foo只是指向函數(shù)對(duì)象的指針
console.log(bar());

因此,我們應(yīng)該如何實(shí)現(xiàn)對(duì)象的深淺復(fù)制?

復(fù)制對(duì)象

在JavaScript中,復(fù)制對(duì)象分為兩種方式,淺復(fù)制和深復(fù)制。

淺復(fù)制沒有辦法去真正的去復(fù)制一個(gè)對(duì)象,而只是保存了對(duì)該對(duì)象的引用;而深復(fù)制可以實(shí)現(xiàn)真正的復(fù)制一個(gè)對(duì)象。

淺復(fù)制

在ES6中,Object對(duì)象新增了一個(gè)assign方法,可以實(shí)現(xiàn)對(duì)象的淺復(fù)制。這里談?wù)凮bject.assign方法的具體用法,因?yàn)樯院髸?huì)分析jQuery的extend方法,實(shí)現(xiàn)的原理同Object.assign方法差不多

Object.assign的第一個(gè)參數(shù)是目標(biāo)對(duì)象,可以跟一或多個(gè)源對(duì)象作為參數(shù),將源對(duì)象的所有可枚舉([[emuerable]] === true)復(fù)制到目標(biāo)對(duì)象。這種復(fù)制屬于淺復(fù)制,復(fù)制對(duì)象時(shí)只是包含對(duì)該對(duì)象的引用。Object.assign(target, [source1, source2, ...])

  • 如果目標(biāo)對(duì)象與源對(duì)象有同名屬性,則后面的屬性會(huì)覆蓋前面的屬性
  • 如果只有一個(gè)參數(shù),則直接返回該參數(shù)。即Object.assign(obj) === obj
  • 如果第一個(gè)參數(shù)不是對(duì)象,而是基本數(shù)據(jù)類型(Null、Undefined除外),則會(huì)調(diào)用對(duì)應(yīng)的基本包裝類型
  • 如果第一個(gè)參數(shù)是Null和Undefined,則會(huì)報(bào)錯(cuò);如果Null和Undefined不是位于第一個(gè)參數(shù),則會(huì)略過(guò)該參數(shù)的復(fù)制

要實(shí)現(xiàn)對(duì)象的淺復(fù)制,可以使用Object.assign方法

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

不過(guò)對(duì)于深復(fù)制來(lái)說(shuō),Object.assign方法無(wú)法實(shí)現(xiàn)

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對(duì)象中e屬性的改變,仍然會(huì)影響到obj對(duì)象

深復(fù)制

在實(shí)際的開發(fā)項(xiàng)目中,前后端進(jìn)行數(shù)據(jù)傳輸,主要是通過(guò)JSON實(shí)現(xiàn)的。JSON全稱:JavaScript Object Notation,JavaScript對(duì)象表示法。

JSON對(duì)象下有兩個(gè)方法,一是將JS對(duì)象轉(zhuǎn)換成字符串對(duì)象的JSON.stringify方法;一個(gè)是將字符串對(duì)象轉(zhuǎn)換成JS對(duì)象的JSON.parse方法。

這兩個(gè)方法結(jié)合使用可以實(shí)現(xiàn)對(duì)象的深復(fù)制。也就是說(shuō),當(dāng)我們需要復(fù)制一個(gè)obj對(duì)象時(shí),可以先調(diào)用JSON.stringify(obj),將其轉(zhuǎn)換為字符串對(duì)象,然后再調(diào)用JSON.parse方法,將其轉(zhuǎn)換為JS對(duì)象。就可以輕松的實(shí)現(xiàn)對(duì)象的深復(fù)制

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

當(dāng)然,使用這種方式實(shí)現(xiàn)深復(fù)制有一個(gè)缺點(diǎn)就是必須給JSON.parse方法傳入的字符串必須是合法的JSON,否則會(huì)拋出錯(cuò)誤

jQuery.extend || jQuery.fn.extend

jQuery.extend對(duì)象,對(duì)使用jQuery超過(guò)一定時(shí)間的朋友來(lái)說(shuō)并不默認(rèn)。這個(gè)$.extend方法可以用來(lái)擴(kuò)展jQuery的全局對(duì)象,而$.fn.extend方法可以用來(lái)擴(kuò)展實(shí)例對(duì)象。fn實(shí)際上是prototype對(duì)象的別名,所以,擴(kuò)展實(shí)例對(duì)象的方法實(shí)際上就是在jQuery原型對(duì)象上添加一些方法。

$.extend方法不僅可以用來(lái)寫jQuery插件,同樣的,它可以用來(lái)實(shí)現(xiàn)對(duì)象的深淺復(fù)制。(使用$.extend與$.fn.extend實(shí)現(xiàn)深淺復(fù)制都可以,唯一的差別就是this的指向性不同)

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

  • 當(dāng)不接受任何參數(shù)時(shí),直接返回一個(gè)空對(duì)象
  • 當(dāng)只有一個(gè)參數(shù)時(shí)(這個(gè)參數(shù)可以任何數(shù)據(jù)類型(Null、Undefined、Boolean、String、Number、Object)),會(huì)返回this對(duì)象,這里會(huì)分為兩種情況。如果用$.extend,會(huì)返回jQuery對(duì)象;如果用$.fn.extend,會(huì)返回jQuery的原型對(duì)象。
  • 當(dāng)接收兩個(gè)參數(shù)時(shí),并且第一個(gè)參數(shù)是Boolean值時(shí),也會(huì)返回一個(gè)空對(duì)象。如果第一個(gè)參數(shù)不是Boolean值,那么會(huì)將源對(duì)象復(fù)制到目標(biāo)對(duì)象
  • 當(dāng)接收三個(gè)參數(shù)以上時(shí),可以分為兩種情況。如果第一個(gè)參數(shù)是Boolean值表示深淺復(fù)制,那么目標(biāo)對(duì)象會(huì)移動(dòng)到第二個(gè)參數(shù),源對(duì)象會(huì)移動(dòng)到第三個(gè)參數(shù)。(目標(biāo)對(duì)象、源對(duì)象和Object.assign方法中的相同)。如果第一個(gè)參數(shù)不是Boolean值,那么用法與Object.assign方法常規(guī)的復(fù)制相同。
  • 在循環(huán)源對(duì)象的過(guò)程中,任何數(shù)據(jù)類型為Null、Undefined或者源對(duì)象是一個(gè)空對(duì)象時(shí),在復(fù)制的過(guò)程中都會(huì)被忽略。
  • 如果源對(duì)象和目標(biāo)對(duì)象具有同名的屬性,則源對(duì)象的屬性會(huì)覆蓋掉目標(biāo)對(duì)象中的屬性。如果同名屬性是一個(gè)對(duì)象的話,則會(huì)在deep=true等其他條件下向目標(biāo)對(duì)象的該同名對(duì)象添加屬性

下面貼出jQuery-2.1.4中jQuery.extend實(shí)現(xiàn)方式的源代碼

jQuery.extend = jQuery.fn.extend = function() {
  var options, name, src, copy, copyIsArray, clone,
    target = arguments[0] || {},
    // 使用||運(yùn)算符,排除隱式強(qiáng)制類型轉(zhuǎn)換為false的數(shù)據(jù)類型
    // 如'', 0, undefined, null, false等
    // 如果target為以上的值,則設(shè)置target = {}
    i = 1,
    length = arguments.length,
    deep = false;

  // 當(dāng)typeof target === 'boolean'時(shí)
  // 則將deep設(shè)置為target的值
  // 然后將target移動(dòng)到第二個(gè)參數(shù),
  if (typeof target === "boolean") {
    deep = target;
    // 使用||運(yùn)算符,排除隱式強(qiáng)制類型轉(zhuǎn)換為false的數(shù)據(jù)類型
    // 如'', 0, undefined, null, false等
    // 如果target為以上的值,則設(shè)置target = {}
    target = arguments[i] || {};
    i++;
  }

  // 如果target不是一個(gè)對(duì)象或數(shù)組或函數(shù),
  // 則設(shè)置target = {}
  // 這里與Object.assign的處理方法不同,
  // assign方法會(huì)將Boolean、String、Number方法轉(zhuǎn)換為對(duì)應(yīng)的基本包裝類型
  // 然后再返回,
  // 而extend方法直接將typeof不為object或function的數(shù)據(jù)類型
  // 全部轉(zhuǎn)換為一個(gè)空對(duì)象
  if (typeof target !== "object" && !jQuery.isFunction(target)) {
    target = {};
  }

  // 如果arguments.length === 1 或
  // typeof arguments[0] === 'boolean', 且存在arguments[1],
  // 這時(shí)候目標(biāo)對(duì)象會(huì)指向this
  // this的指向哪個(gè)對(duì)象需要看是使用$.fn.extend還是$.extend
  if (i === length) {
    target = this;
    // i-- 表示不進(jìn)入for循環(huán)
    i--;
  }

  // 循環(huán)arguments類數(shù)組對(duì)象,從源對(duì)象開始
  for (; i < length; i++) {
    // 針對(duì)下面if判斷
    // 有一點(diǎn)需要注意的是
    // 這里有一個(gè)隱式強(qiáng)制類型轉(zhuǎn)換 undefined == null 為 true
    // 而undefined === null 為 false
    // 所以如果源對(duì)象中數(shù)據(jù)類型為Undefined或Null
    // 那么就會(huì)跳過(guò)本次循環(huán),接著循環(huán)下一個(gè)源對(duì)象
    if ((options = arguments[i]) != null) {
      // 遍歷所有[[emuerable]] === true的源對(duì)象
      // 包括Object, Array, String
      // 如果遇到源對(duì)象的數(shù)據(jù)類型為Boolean, Number
      // for in循環(huán)會(huì)被跳過(guò),不執(zhí)行for in循環(huán)
      for (name in options) {
        // src用于判斷target對(duì)象是否存在name屬性
        src = target[name];

        // 需要復(fù)制的屬性
        // 當(dāng)前源對(duì)象的name屬性
        copy = options[name];

        // 這種情況暫時(shí)未遇到..
        // 按照我的理解,
        // 即使copy是同target是一樣的對(duì)象
        // 兩個(gè)對(duì)象也不可能相等的..
        if (target === copy) {
          continue;
        }

        // if判斷主要用途:
        // 如果是深復(fù)制且copy是一個(gè)對(duì)象或數(shù)組
        // 則需要遞歸jQuery.extend(),
        // 直到copy成為一個(gè)基本數(shù)據(jù)類型為止
        if (deep && copy && (jQuery.isPlainObject(copy) || (copyIsArray = jQuery.isArray(copy)))) {
          // 深復(fù)制
          if (copyIsArray) {
            // 如果是copy是一個(gè)數(shù)組
            // 將copyIsArray重置為默認(rèn)值
            copyIsArray = false;
            // 如果目標(biāo)對(duì)象存在name屬性且是一個(gè)數(shù)組
            // 則使用目標(biāo)對(duì)象的name屬性,否則重新創(chuàng)建一個(gè)數(shù)組,用于復(fù)制
            clone = src && jQuery.isArray(src) ? src : [];

          } else {
            // 如果目標(biāo)對(duì)象存在name屬性且是一個(gè)對(duì)象
            // 則使用目標(biāo)對(duì)象的name屬性,否則重新創(chuàng)建一個(gè)對(duì)象,用于復(fù)制
            clone = src && jQuery.isPlainObject(src) ? src : {};
          }

          // 因?yàn)樯顝?fù)制,所以遞歸調(diào)用jQuery.extend方法
          // 返回值為target對(duì)象,即clone對(duì)象
          // copy是一個(gè)源對(duì)象
          target[name] = jQuery.extend(deep, clone, copy);

        } else if (copy !== undefined) {
          // 淺復(fù)制
          // 如果copy不是一個(gè)對(duì)象或數(shù)組
          // 那么執(zhí)行elseif分支
          // 在elseif判斷中如果copy是一個(gè)對(duì)象或數(shù)組,
          // 但是都為空的話,排除這種情況
          // 因?yàn)楂@取空對(duì)象的屬性會(huì)返回undefined
          target[name] = copy;
        }
      }
    }
  }

  // 當(dāng)源對(duì)象全部循環(huán)完畢之后,返回目標(biāo)對(duì)象
  return target;
};   

因此,可以針對(duì)分析過(guò)后的源碼,給出一些例子

let obj1 = $.extend();
console.log(obj1); // 返回一個(gè)空對(duì)象 {}

let obj2 = $.extend(undefined);
console.log(obj2); //返回jQuery對(duì)象,Object.assign傳入undefined會(huì)報(bào)錯(cuò)

let obj3 = $.extend('123');
console.log(obj3); // 返回jQuery對(duì)象,Object.assign傳入'123'會(huì)返回字符串的String對(duì)象

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}
// 默認(rèn)情況下,復(fù)制方式都是淺復(fù)制
// 如果只需要淺復(fù)制,不傳入deep參數(shù)也可以
// 淺復(fù)制時(shí),obj4對(duì)象中的d屬性只是指向數(shù)組對(duì)象的指針

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}
// 會(huì)略過(guò)空對(duì)象或Undefined、Null值

let obj7 = $.extend(true, target, source1, source2);
console.log(obj7); // {a: 123, b: 456, d: Array(3), c: 789}
// 這里target對(duì)象有b屬性,源對(duì)象source1也有b屬性
// 此時(shí)源對(duì)象的b屬性會(huì)覆蓋目標(biāo)對(duì)象的b屬性
// 這里deep=true,屬于深復(fù)制
// 當(dāng)name=d時(shí),會(huì)遞歸調(diào)用$.extend, 直到它的屬性對(duì)應(yīng)的屬性值全部為基本數(shù)據(jù)類型
// 源對(duì)象的改變不會(huì)影響到obj7對(duì)象

JavaScript 復(fù)制對(duì)象

因此,可以根據(jù)$.extend方法,寫出一個(gè)通用的實(shí)現(xiàn)對(duì)象深淺復(fù)制的函數(shù),copyObject函數(shù)唯一的不同就是當(dāng)i === arguments.length屬性時(shí),copyObject函數(shù)直接返回了target對(duì)象

function copyObject () {
  let i = 1,
    target = arguments[0] || {},
    deep = false,
    length = arguments.length,
    name, options, src, copy,
    copyIsArray, clone;

  // 如果第一個(gè)參數(shù)的數(shù)據(jù)類型是Boolean類型
  // target往后取第二個(gè)參數(shù)
  if (typeof target === 'boolean') {
    deep = target;
    // 使用||運(yùn)算符,排除隱式強(qiáng)制類型轉(zhuǎn)換為false的數(shù)據(jù)類型
    // 如'', 0, undefined, null, false等
    // 如果target為以上的值,則設(shè)置target = {}
    target = arguments[1] || {};
    i++;
  }

  // 如果target不是一個(gè)對(duì)象或數(shù)組或函數(shù)
  if (typeof target !== 'object' && !(typeof target === 'function')) {
    target = {};
  }

  // 如果arguments.length === 1 或
  // typeof arguments[0] === 'boolean',
  // 且存在arguments[1],則直接返回target對(duì)象
  if (i === length) {
    return target;
  }

  // 循環(huán)每個(gè)源對(duì)象
  for (; i < length; i++) {
    // 如果傳入的源對(duì)象是null或undefined
    // 則循環(huán)下一個(gè)源對(duì)象
    if (typeof (options = arguments[i]) != null) {
      // 遍歷所有[[emuerable]] === true的源對(duì)象
      // 包括Object, Array, String
      // 如果遇到源對(duì)象的數(shù)據(jù)類型為Boolean, Number
      // for in循環(huán)會(huì)被跳過(guò),不執(zhí)行for in循環(huán)
      for (name in options) {
        // src用于判斷target對(duì)象是否存在name屬性
        src = target[name];
        // copy用于復(fù)制
        copy = options[name];
        // 判斷copy是否是數(shù)組
        copyIsArray = Array.isArray(copy);
        if (deep && copy && (typeof copy === 'object' || copyIsArray)) {
          if (copyIsArray) {
            copyIsArray = false;
            // 如果目標(biāo)對(duì)象存在name屬性且是一個(gè)數(shù)組
            // 則使用目標(biāo)對(duì)象的name屬性,否則重新創(chuàng)建一個(gè)數(shù)組,用于復(fù)制
            clone = src && Array.isArray(src) ? src : [];
          } else {
            // 如果目標(biāo)對(duì)象存在name屬性且是一個(gè)對(duì)象
            // 則使用目標(biāo)對(duì)象的name屬性,否則重新創(chuàng)建一個(gè)對(duì)象,用于復(fù)制
            clone = src && typeof src === 'object' ? src : {};
          }
          // 深復(fù)制,所以遞歸調(diào)用copyObject函數(shù)
          // 返回值為target對(duì)象,即clone對(duì)象
          // copy是一個(gè)源對(duì)象
          target[name] = copyObject(deep, clone, copy);
        } else if (copy !== undefined){
          // 淺復(fù)制,直接復(fù)制到target對(duì)象上
          target[name] = copy;
        }
      }
    }
  }
  // 返回目標(biāo)對(duì)象
  return target;   
}

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI