JS 拷贝的"分身术":深拷贝 vs 浅拷贝

在 JavaScript 中,因为对象和数组是引用类型,所以当我们直接将它们赋值给另一个变量时,实际上是将它们的引用地址复制了一份。这样,当我们对其中一个变量进行修改时,另一个变量也会受到影响。因此,为了避免这种情况,我们通常需要使用拷贝方法来复制一个对象或数组的值并创建一个新的副本。本文将来介绍浅拷贝和深拷贝的概念以及它们的应用场景

浅拷贝

定义:浅拷贝只复制对象的第一层属性,如果属性是基本类型,则赋值其值;如果是引用类型则赋值其内存地址

特点

  • 新对象和原对象共享引用类型的属性
  • 修改拷贝对象的引用类型属性会影响原对象
  • 实现简单,性能较好

浅拷贝常见实现方法

1、Object.create(obj)

Object.create() 方法可以用于创建一个新对象,并将原对象作为新对象的原型。这样,新对象就可以访问原对象的所有属性和方法

js 复制代码
const obj1 = { name: '张三', age: 18 };
const obj2 = Object.create(obj1);
console.log(obj2.name); // 张三

obj1.name = '李四'
console.log(obj2.name); //李四

2、Object.assign({},obj)

Object.assign({}, obj) 是实现对象浅拷贝的标准方法,它通过将源对象的可枚举属性复制到目标对象(这里是空对象 {})来实现拷贝。

js 复制代码
const originalObj = { a: 1, b: { c: 2 } };
const copiedObj = Object.assign({}, originalObj);

console.log(copiedObj); // { a: 1, b: { c: 2 } }
console.log(copiedObj === originalObj); // false (新对象)

3、[].concat(arr)

[].concat(arr) 是实现数组浅拷贝的一种传统方法,它通过连接空数组和目标数组来创建一个新数组。

js 复制代码
const arr = [1, 2, 3];
const newArr = [].concat(arr);

console.log(newArr); // [1, 2, 3]
console.log(newArr === arr); // false (是不同的数组实例)

4、[...arr]

[...arr] 是 ES6 引入的展开运算符(Spread Operator) 用于数组的浅拷贝,这是现代 JavaScript 中最简洁的数组浅拷贝方式之一。

js 复制代码
const originalArr = [1, 2, 3];
const copiedArr = [...originalArr];

console.log(copiedArr); // [1, 2, 3]
console.log(copiedArr === originalArr); // false (新数组)

5、arr.slcie()

arr.slice() 是 JavaScript 数组的一个内置方法,用于创建数组的浅拷贝(shallow copy)

js 复制代码
const arr = [1, 2, 3];
const newArr = arr.slice();
console.log(newArr);//[1,2,3]
console.log(newArr === arr); // false (是新数组)

深拷贝

定义:深拷贝会递归复制对象的所有层级,创建一个完全独立的副本,新旧对象不会共享任何引用。

特点

  • 无论修改原始对象的顶层还是嵌套子对象,均不会影响拷贝后的对象
  • 需要更多内存和计算资源(尤其是复杂对象)

深拷贝常见实现方法

1、JSON.parse(JSON.stringify())

JSON.parse(JSON.stringify())是最简单的深拷贝实现方法但其有局限性,无法处理undefinedfunctionSymbolDateRegExpMapSet 等。

js 复制代码
const obj = { a: 1, b: { c: 2 } };
const deepCopy = JSON.parse(JSON.stringify(obj));

obj.b.c = 99;
console.log(deepCopy.b.c); // 输出 2(未被修改)

2、structuredClone() structuredClone()是浏览器和Node.js提供的原生深拷贝方案比JSON.parse(JSON.stringify())更强大且安全。

js 复制代码
const obj = { a: 1, b: new Date(), c: new Set([1, 2]) };
const deepCopy = structuredClone(obj);

obj.c.add(3);
console.log(deepCopy.c); // 输出 Set { 1, 2 }(未被修改)

浅拷贝和深拷贝关键区别

关键区别总结

特性 浅拷贝 深拷贝
复制层级 仅顶层 所有嵌套层级
子对象引用 共享(指向同一内存地址) 独立(新内存地址)
性能 慢(递归复制)
适用场景 简单对象 需要完全隔离的复杂对象

面试题:手搓一个深浅拷贝

浅拷贝:

js 复制代码
let obj = {
    name: '困困',
    age: 18,
    like: {
        a: '唱',
        b: '跳',
        c: 'rap'
    }
}
function shallowCopy(obj) {
    let newObj = {}
    for (let key in obj) {
        if (obj.hasOwnProperty(key)) { //判断key是不是obj显示拥有的
            newObj[key] = obj[key]
        }
    }
    return newObj
}
console.log(shallowCopy(obj))

深拷贝

js 复制代码
    let obj = {
        name: '困困',
        age: 18,
        like: {
            a: '唱',
            b: '跳',
            c: 'rap'
        }
    }
    function deepCopy(obj) {
        let newObj = {}
        for (key in obj) {
            if (obj.hasOwnProperty(key)) {
                // 先判断obj[key]值的类型,如果是原始类型,直接赋值,如果是引用类型,递归调用深拷贝函数
                if(typeof obj[key]==='object'&&obj[key]!=null){
                    newObj[key]=deepCopy(obj[key])
                }
                else{
                    newObj[key] = obj[key]
                }          
            }

        }
        return newObj
    }
    let obj2=deepCopy(obj)
    console.log(obj2)
相关推荐
好好研究3 小时前
使用JavaScript实现轮播图的自动切换和左右箭头切换效果
开发语言·前端·javascript·css·html
伍哥的传说6 小时前
Radash.js 现代化JavaScript实用工具库详解 – 轻量级Lodash替代方案
开发语言·javascript·ecmascript·tree-shaking·radash.js·debounce·throttle
前端程序媛-Tian7 小时前
【dropdown组件填坑指南】—怎么实现下拉框的位置计算
前端·javascript·vue
iamlujingtao7 小时前
js多边形算法:获取多边形中心点,且必定在多边形内部
javascript·算法
嘉琪0017 小时前
实现视频实时马赛克
linux·前端·javascript
爱分享的程序员8 小时前
前端面试专栏-前沿技术:30.跨端开发技术(React Native、Flutter)
前端·javascript·面试
超级土豆粉8 小时前
Taro 位置相关 API 介绍
前端·javascript·react.js·taro
草履虫建模8 小时前
RuoYi-Vue 项目 Docker 容器化部署 + DockerHub 上传全流程
java·前端·javascript·vue.js·spring boot·docker·dockerhub
拾光拾趣录9 小时前
前端灵魂拷问:10道题
前端·面试
阿丽塔~10 小时前
【vue3+vue-pdf-embed】实现PDF+图片预览
javascript·vue.js·pdf