JavaScript 中的浅拷贝 (shallow copy) 和深拷贝 (deep copy) 差别是什么? 要如何实践?

2024年1月3日

💎 加入 E+ 成長計畫 與超過 300+ 位軟體工程師一同在社群中成長,並且獲得更多的軟體工程學習資源

在 JavaScript 复制值时,当复制的是非原始型别(primitive type) 的资料型别时,例如:物件(object)、数组(array) 等,会遇到浅拷贝(shallow copy) 和深拷贝(deep copy) 的差异。在面试时被问到这两者的差异,你会怎么回答? 如果要你当场手写深拷贝,你会怎么写? 假如不确定的话,就一起来读这篇吧。

比较浅拷贝 (shallow copy) 和深拷贝 (deep copy)

浅拷贝是指复制值时,满足物件 A 与物件 B 不同,但物件 A 与 物件 B 有相同的属性,并且属性的原型链相同。

而深拷贝则是指在拷贝时,物件 A 与物件 B 不同,两者在原型链上仅是结构相同,但其属性实际的地址不同。在拷贝值时,有可能会遇到变数是多层的情境,例如是一个物件里还有物件,深拷贝的定义会是每一层的值都不会共享址 (reference)。

这样听起来可能比较抽象,具体来说,以 lodash 这个套件提供的效用函式为例,有分成 clonecloneDeep 两种不同效用函式,clone 只用于浅拷贝(第一层拷贝),但 cloneDeep 可用于深拷贝。下面的例子说明两者的区别:

// lodash 的浅拷贝 clone
var objects = [{ a: 1 }, { b: 2 }];
var shallow = _.clone(objects);
console.log(objects === shallow); // false
console.log(shallow[0] === objects[0]); // true

// lodash 的深拷贝 cloneDeep
var objects = [{ a: 1 }, { b: 2 }];
var deep = _.cloneDeep(objects);
console.log(objects === deep); // false
console.log(deep[0] === objects[0]); // false

在说明完浅拷贝与深拷贝的差别后,面试中常见的接续问题是「手写」浅拷贝与深拷贝。假如你不确定怎么手写这两种拷贝方式,可以继续往下看。

浅拷贝 (shallow copy)

方法一:手动复制值

let objA = {
  a: 1,
  b: { c: 3 },
};

let objB = { a: objA.a, b: objA.b };

console.log(objA === objB); // false
console.log(objA.b === objB.b); // true, 第二层的物件还是指向相同位置

方法二:使用 spread syntax

let objA = {
  a: 1,
  b: { c: 3 },
};

let objB = { ...objA };

console.log(objA === objB); // false
console.log(objA.b === objB.b); // true, 第二层的物件还是指向相同位置

方法三:使用 Object.assign

let objA = {
  a: 1,
  b: { c: 3 },
};

let objB = Object.assign({}, objA);

console.log(objA === objB); // false
console.log(objA.b === objB.b); // true, 第二层的物件还是指向相同位置

深拷贝 (deep copy)

方法一:使用 JSON.parse(JSON.stringify(...))

这个作法是先将物件用 JSON.stringify 序列化为 string,再透过 JSON.parse 转换回物件。要特别注意,这做法只能用于可序列化的物件,有些无法序列化的物件例如:function、HTML 的元素,这些是无法序列化的,所以执行前,需要先确认是否可以序列化,否则在执行JSON.stringify 时会失败。

let objA = {
  a: 1,
  b: { c: 3 },
};

function deepCopy(item) {
  return JSON.parse(JSON.stringify(item));
}

let objB = deepCopy(objA);

console.log(objA === objB); // false
console.log(objA.b === objB.b); // false

方法二:使用 structuredClone(value)

针对可序列化的物件,有另外一种透过 JavaScript 内建的方法达成深拷贝。这种方法是 structuredClone(value),用法如下。

let objA = {
  a: 1,
  b: { c: 3 },
};

let objB = structuredClone(objA);

console.log(objA === objB); // false
console.log(objA.b === objB.b); // false

方法三:考虑多重情况的递回式深拷贝

通常在面试中,用上述两种方式,可能会被面试官追问说,如果不用这种现成的方法,要如何手写。以下的写法是透过递回的方式,来进行深拷贝。

function deepClone(obj, cache = new WeakMap()) {
  if (cache.has(obj)) {
    return cache.get(obj);
  }

  if (obj === null || typeof obj !== "object" || typeof value === "function") {
    return obj;
  }

  if (obj instanceof Date) return new Date(obj);
  if (obj instanceof RegExp) return new RegExp(obj);

  const result = Array.isArray(obj)
    ? []
    : Object.create(Object.getPrototypeOf(obj));

  cache.set(obj, result);

  for (const key of Reflect.ownKeys(obj)) {
    const value = obj[key];
    result[key] = deepClone(value, cache);
  }

  return result;
}
🧵 如果你想收到最即時的內容更新,可以在 FacebookInstagram 上追蹤我們