博客地址:https://ainyi.com/72html
JavaScript 程序中,对于简单的数字、字符串能够经过 = 赋值拷贝
可是对于数组、对象、对象数组的拷贝,就有浅拷贝和深拷贝之分es6
浅拷贝就是当改变了拷贝后的数据,原数据也会相应改变json
来讲说深拷贝数组
不推荐此方法函数
let a = [1, 2, 3] let b = [] for (let val of a) { b.push(val) } b.push(4) a // [1, 2, 3] b // [1, 2, 3, 4]
数组方法 slice() 可从已有的数组中返回选定的元素
那么设置为 0,就是返回整个数组插件
let a = [1, 2, 3] let b = a.slice(0) b.push(4) a // [1, 2, 3] b // [1, 2, 3, 4]
数组方法 concat() 链接一个或多个数组,并返回一个副本
那么不设置参数,就返回本数组code
let a = [1, 2, 3] let b = a.concat() b.push(4) a // [1, 2, 3] b // [1, 2, 3, 4]
let a = [1, 2, 3] let b = [...a] b.push(4) a // [1, 2, 3] b // [1, 2, 3, 4]
let a = [1, 2, 3] let b = Array.from(a) b.push(4) a // [1, 2, 3] b // [1, 2, 3, 4]
Object.assign(target, obj)htm
let a = { name: 'krry' } let b = Object.assign({}, a) b.name = 'lily' a // { name: 'krry' } b // { name: 'lily' }
注意使用 assign() 有以下特色:对象
let a = { name: 'krry' } let b = { ...a } b.name = 'lily' a // { name: 'krry' } b // { name: 'lily' }
以上是简单数组、对象的深拷贝方法,可是对于二维数组、对象数组、对象里包含对象,以上方法均达不到深拷贝方法
以上只能达到数组、对象的第一层的==深拷贝==,对于里面的数组或对象属性则是==浅拷贝==,由于里面的内存地址只是拷贝了一份,但都是指向==同一个地址==
因此当改变数组、对象里的数组元素或对象,原数据依然会改变继承
使用 JSON.parse(JSON.stringify(obj))
let a = [1, [2, {aa: 2}, [4]], {aa: 5, cc: { dd: 6 }}] let b = JSON.parse(JSON.stringify(a)) // 完美
经过 JSON.stringify 实现深拷贝有几点要注意
function deepCopy(obj) { let result = Array.isArray(obj) ? [] : {}; for (let key in obj) { if (obj.hasOwnProperty(key)) { if (typeof obj[key] === 'object') { result[key] = deepCopy(obj[key]); // 递归复制 } else { result[key] = obj[key]; } } } return result; }
使用 lodash 插件的深拷贝方法
// 官方例子 var objects = [{ 'a': 1 }, { 'b': 2 }]; var deep = _.cloneDeep(objects); console.log(deep[0] === objects[0]); // => false
传送门:https://www.lodashjs.com/docs/4.17.5.html#cloneDeep
博客地址:https://ainyi.com/72