1. 程式人生 > >如何深拷貝一個物件陣列?

如何深拷貝一個物件陣列?

一、背景

某個專案裡,存在一個物件陣列,我用 lodash 的 filter() 函式,分別生成了 A、B 兩個新的物件陣列,但我遍歷了 B 陣列,改造裡面的每一個物件,沒想到引起 A 陣列的裡物件發生了變化,引發了錯誤。

這是一個基礎的,對引用型別——物件沒有使用深拷貝的問題,我疏忽了,特此記錄下。

二、例子

1、淺拷貝

const _ = require('lodash');

let one_brand = [
    {name: 'A', count: 1, value: Infinity},
    {name: 'B', count: 2},
]

// 淺拷貝
let two_brand = one_brand;
 
console.log("改變前:")
console.log(one_brand) 
console.log(two_brand) 

two_brand[1].count = 0;

console.log("改變後:")
console.log(one_brand) 
console.log(two_brand) 

return:

改變前:
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 2 } ]
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 2 } ]
改變後:
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 0 } ]
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 0 } ]

你會發現改變了 two_brand 的一個物件,one_brand 的那個對應的物件也改變了。這樣不行。

2、深拷貝

const _ = require('lodash');

let one_brand = [
    {name: 'A', count: 1, value: Infinity},
    {name: 'B', count: 2},
]

// 深拷貝
// let two_brand = one_brand.map(o => Object.assign({}, o));
// or 更簡單的
let two_brand = one_brand.map(o => ({...o}));
 
console.log("改變前:")
console.log(one_brand) 
console.log(two_brand) 

two_brand[1].count = 0;

console.log("改變後:")
console.log(one_brand) 
console.log(two_brand) 

return:

改變前:
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 2 } ]
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 2 } ]
改變後:
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 2 } ]
[ { name: 'A', count: 1, value: Infinity },
  { name: 'B', count: 0 } ]

3、拓展

其實網上還有一種方法:

let two_brand = JSON.parse(JSON.stringify(one_brand))

這種方法存在很大的問題。雖然他在 stackoverflow 是得票最多的一個答案。(https://stackoverflow.com/questions/597588/how-do-you-clone-an-array-of-objects-in-javascript

它的主要缺點是,只限於處理可被 JSON.stringify() 編碼的值。

JSON.stringify() 將編碼 JSON 支援的值。包含 Boolean,Number,String,以及物件,陣列。其他任何內容都將被特殊處理。

undefinedFunctionSymbol 時,它被忽略掉
InfinityNaN 會被變成 null
Date 物件會被轉化為 String (預設呼叫date.toISOString())

問:為什麼JSON.stringify() 編碼 JSON 支援的值那麼少呢?

因為JSON是一個通用的文字格式,和語言無關。設想如果將函式定義也stringify的話,如何判斷是哪種語言,並且通過合適的方式將其呈現出來將會變得特別複雜。特別是和語言相關的一些特性,比如JavaScript中的Symbol。

來源: https://www.cnblogs.com/xjnotxj/p/9810534.html

 

鄭州人工授精醫院

鄭州包皮手術多少錢

鄭州不孕不育醫院哪家好

鄭州男科哪家好