js浅拷贝及深拷贝的几种方法

本文详细介绍了JavaScript中对象和数组的拷贝方法,包括浅拷贝的Object.assign()和解构赋值,以及深拷贝的JSON.parse(JSON.stringify())和递归实现。通过实例展示了各种拷贝方式在处理复杂数据结构时的行为,并分析了它们的适用场景和限制。

一个项目开发中经常会用到需要复制一个对象或者数组,但是却不能改变原始对象,所以就要用到拷贝,拷贝又分深拷贝和浅拷贝,今天列举一下几种拷贝形式。

一、浅拷贝

(1) Object.assign()

Object.assign我们经常会用到合并对象,当然利用Object.assign性质我们也可以实现对象的拷贝。

var obj1 = {a: 1, b: 2}
 
var obj2 = Object.assign({}, obj1)
 
obj2.a = 4
 
console.log(obj1, obj2)

结果如图:

这里要注意的是Object.assign第一个参数必须是个空对象

(2) 解构赋值

var obj1 = {a: 1, b: 2}
 
var obj2 = {...obj1}
 
obj2.a = 4
 
console.log(obj1, obj2)

结果如图:

这里一样可以实现之前上面的结果。

但这两种拷贝有一个问题就是只能赋值一层,假设我们有如下数据结构

var obj1 = [{
    name: '臧三',
    childs: ['小明', '小芳']
}]
 
var obj2 = [...obj1]
 
obj2[0].childs = []
 
console.log(obj1, obj2)

我们会发现打印出的结果如下:

上图可看出obj1,obj2 的结果均变了,这并不是我们想要的结果,所以我们要用到深拷贝。

二、深拷贝

(1) 利用json.stringify

var obj1 = [{
    name: '臧三',
    childs: ['小明', '小芳']
}]
 
var obj2 = JSON.parse(JSON.stringify(obj1))
 
obj2[0].childs = []
 
console.log(obj1, obj2)

结果如下:

这样的话就能达到我们想要的结果。

这种方法简单,但也有弊端,看看下面的数据结构:

var obj1 = [{
    name: '臧三',
    childs: ['小明', '小芳'],
    fn: function() {},
    age: undefined
}]
 
var obj2 = JSON.parse(JSON.stringify(obj1))
 
obj2[0].childs = []
 
console.log(obj1, obj2)

结果:

我们从结果中发现,值为undefined,或者function的时候并不会拷贝过来。

(2) 利用递归来实现一个方法进行拷贝

var obj1 = [{
    name: '臧三',
    childs: ['小明', '小芳'],
    fn: function() {},
    age: undefined
}]
 
var obj2 = extend(obj1)
 
obj2[0].childs = []
 
console.log(obj1, obj2)
 
function extend(data) {
    if (typeof data === 'object' && data) {
        let val = typeof data.length === 'number' ? [] : {}
        for(let i in data) {
            val[i] = extend(data[i])
        }
        return val
    } else {
        return data
    }
}

结果:

这样我们就能把所有值都拷贝过来。

### JavaScript深拷贝浅拷贝的概念 在 JavaScript 中,当处理复杂数据结构如对象或数组时,区分深拷贝浅拷贝至关重要。对于基本类型的变量(如数字、布尔值),赋值操作即创建了一个新的独立副本;但对于引用类型的数据(如对象、数组),情况则不同。 #### 浅拷贝 浅拷贝是指创建一个新对象,这个新对象拥有原始对象属性值的一份精确拷贝。如果这些属性也是对象,则只会复制它们的引用而不是其实际内容。因此,在修改新对象内部某个嵌套对象的内容时会影响到原对象[^3]。 常见实现方式有: - **Object.assign()** 此方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象,并返回该目标对象。 ```javascript const target = { a: 1 }; const source = { b: 2, c: { d: 3 } }; const result = Object.assign({}, target, source); ``` - **扩展运算符 (`...`)** 类似于 `Object.assign()` 的功能,可以用来简化语法。 ```javascript const obj1 = { propA: 'value' }; const shallowCopy = { ...obj1 }; ``` - **Array 方法** 对于数组而言,还可以通过一些特定的方法来完成浅层复制工作,比如 `.slice()` `.concat()` 等。 ```javascript const arr = ['a', 'b']; const newArr = arr.slice(); ``` #### 深拷贝 相比之下,深拷贝不仅会复制最外层的对象本身,还会递归地对其所有的子节点进行同样的操作直到整个树形结构都被完全克隆出来。这意味着即使是最深层级上的任何改变都不会反映回原来的对象上。 以下是几种常见的深拷贝技术: - **JSON 序列化/反序列化** 利用 JSON.stringify 可以把任意复杂的 JS 数据转换成字符串形式然后再解析回来得到全新的实例。不过需要注意此法无法处理函数、undefined 值等情况。 ```javascript function deepClone(obj) { try { return JSON.parse(JSON.stringify(obj)); } catch (e) { console.error('Deep clone failed:', e.message); throw new Error(e); } } ``` - **递归算法** 编写自定义逻辑逐层遍历并重建每一个组成部分直至结束。 ```javascript function isObject(item) { return item && typeof item === 'object'; } function deepCloneRecursive(source) { if (!isObject(source)) return source; let output; Array.isArray(source) ? (output = []) : (output = {}); for (let key in source) { if (source.hasOwnProperty(key)) { output[key] = isObject(source[key]) && !ArrayBuffer.isView(source[key]) ? deepCloneRecursive(source[key]) : source[key]; } } return output; } ``` - **第三方库支持** 使用像 Lodash 这样的工具包提供了更强大灵活的功能集,能够更好地应对各种特殊情况下的需求。 ```javascript import _ from 'lodash'; const clonedData = _.cloneDeep(originalData); ```
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值