js中的深淺拷貝

晁州發表於2018-04-02

js中的深淺拷貝

js中有深拷貝、淺拷貝一說,所謂的深淺拷貝是針對value型別為引用型別(函式、物件、陣列)而言的,大概理解的就是:

淺拷貝:
    拷貝出的物件c和原始物件o,c和o在key對應的value為引用型別時,其指向同一塊記憶體地址,修改一個必然影響另一個。

舉個淺拷貝的例子:

var shallowCopy = function (src) {
  var dst = {}
  for (const key in src) {
    if (src.hasOwnProperty(key)) {
      dst[key] = src[key];
    }
  }
  return dst
}

var obj = { a: 1, arr: [2, 3] }
var shallowObj = shallowCopy(obj)
console.log(shallowObj) // { a: 1, arr: [ 2, 3 ] }

// 淺拷貝指向同一地址,修改shallowObj的arr則obj.arr也被修改,a屬性是值型別不存在該問題
shallowObj.arr[1] = 5;
console.log(obj)   // { a: 1, arr: [ 2, 5 ] }
shallowObj.a = 22

// { a: 1, arr: [ 2, 5 ] } { a: 22, arr: [ 2, 5 ] }
console.log(obj, shallowObj)   

var obj2 = { a: 1, t: { b: 3, c: 4, d: { e: 6 } } }
var shallowObj2 = shallowCopy(obj2)
// { a: 1, t: { b: 3, c: 4, d: { e: 6 } } }
console.log(shallowObj2)
深拷貝:
    深拷貝出的物件和原物件是完全分開的記憶體地址,不存在修改一個也修改了另一個的問題。

深拷貝的簡單實現:

var china = {
  nation: '中國',
  birthplaces: ['北京', '上海', '廣州'],
  skincolr: 'yellow',
  friends: ['sk', 'ls']
}
//深複製,要想達到深複製就需要用遞迴
function deepCopy(o, c) {
  var c = c || {}
  for (var i in o) {
    if (typeof o[i] === 'object') {
      //要考慮深複製問題了
      if (o[i].constructor === Array) {
        //這是陣列
        c[i] = []
      } else {
        //這是物件
        c[i] = {}
      }
      deepCopy(o[i], c[i])
    } else {
      c[i] = o[i]
    }
  }
  return c
}
var result = { name: 'result' }
result = deepCopy(china, result)
result.friends.push('slj')
/**
{ nation: '中國',
  birthplaces: [ '北京', '上海', '廣州' ],
  skincolr: 'yellow',
  friends: [ 'sk', 'ls' ] }
{ name: 'result',
  nation: '中國',
  birthplaces: [ '北京', '上海', '廣州' ],
  skincolr: 'yellow',
  friends: [ 'sk', 'ls', 'slj' ] }
 */
console.dir(china)
console.dir(result)

其次,深拷貝使用JSON.parse(JSON.stringify(src))也可以簡單實現,只是該實現破壞了原型鏈;lodash、jquery提供了相應的工具函式來實現深淺拷貝,就不記錄了。

相關文章