彻底理解js中的深浅拷贝

数据类型

说到深浅拷贝就不得不说到 js 的数据类型,在 js 中有两大数据类型基本数据类型,引用数据类型。 基本数据类型是存储在内存中,引用数据类型存储在内存中,引用数据类型的变量是一个指向堆内存的的引用,是一个地址值,存在栈中。

浅拷贝

浅拷贝是指创建一份跟原始数据值相同的复制品;规则额是这样的,如果是基本数据类型就拷贝基本类型的值,如果是引用数据类型就拷贝内存地址。也可以这么说浅拷贝就是拷贝一层,深层次的引用类型数据就共享内存地址。 用原生 js 实现一个浅拷贝如下:

js 复制代码
function shallowClone(obj) {
  const shallowObj = {};
  for (const key in obj) {
    if (obj.hasOwnProperty(key)) {
      shallowObj[key] = obj[key];
    }
  }
  return shallowObj;
}

在 js 中原生提供了一些方法供我们实现浅拷贝

  1. Object.assign()
  2. Array.prototype.slice()
  3. Array.prototype.concat()
  4. es6 提供的拓展运算符

看一些例子

Object.assign()

js 复制代码
const obj = {
  name: "zhangsan",
  age: 18,
  hobby: ["apple", "banana"],
  address: {
    province: "省",
    city: "市",
    area: "区",
  },
  say: function () {
    console.log("hello");
  },
};

const newObj = Object.assign({}, obj);

Array.prototype.slice()

js 复制代码
const arr = [1,2,3],
const newArr = arr.slice(0)
newArr[0] = 2
// arr [1,2,3]
// newArr [2,2,3]

Array.prototype.concat()

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

ES6 扩展运算符

js 复制代码
const arr = [1, 2, 3];
const newArr = [...arr];
newArr[0] = 2;
// arr [1,2,3]
// newArr [2,2,3]
// 对象同理
const obj = { a: 1, b: 2 };
const newObj = { ...obj };
newObj[a] = 2;
// obj { a: 1, b: 2 }
// newObj { a: 2, b: 2 }

深拷贝

对于基本数据类型来说深拷贝浅拷贝都是一样的,对于引用数据类型来说,深拷贝就是新开一块内存空间,存放的是一份和原始值一样的数据但是内存地址是不同的,所以修改新对象的属性并不会改变原始对象的属性。

常用的深拷贝方法

  1. lodash 中的 cloneDeep
  2. JSON.stringify() 该方式有很大弊端,当遇到函数、undefined、symbol 时会无法复制,且无法解决循环引用
  3. js 原生手写递归深拷贝函数

原生实现一个深拷贝函数

js 复制代码
function cloneDeep(obj, hash = new WeakMap()) {
  // 如果是null直接return
  if (obj === null) return obj;
  // Date
  if (obj instanceof Date) return new Date(obj);
  // Reg
  if (obj instanceof RegExp) return new RegExp(obj);
  // 如果是基本属性类型或者函数直接return
  if (typeof obj !== "object") return obj;
  // 如果是对象且缓存中有值,直接从缓存中取值
  if (hash.get(obj)) return hash.get(obj);
  // 以obj原型的constructor创建一个新对象,因为它指向当前类本身
  const cloneObj = new obj.constructor();
  hash.set(obj, cloneObj);
  for (const key in obj) {
    if (obj.hasOwnProperty(key)) {
      // 递归
      cloneObj[key] = cloneDeep(obj[key], hash);
    }
  }
  return cloneObj;
}
相关推荐
江畔柳前堤2 小时前
github实战指南07-CLI 与高级技巧
前端·人工智能·chrome·深度学习·github·caffe·issue
kisdiem3 小时前
ReAct:让大模型一边推理,一边行动
前端·react.js·前端框架
西部荒野子3 小时前
JS 如何跑进两个原生世界
前端
RANxy3 小时前
AntV 入门系列第一篇:从零开始的数据可视化之旅
前端
小小小小宇3 小时前
前端 WebRTC 全解析与应用
前端
华玥4 小时前
优化滚动列表,使用虚拟滚动
前端
小小小小宇4 小时前
前端 WebAssembly 全解析与应用
前端
huangdong_4 小时前
京东商品图片视频批量下载与m3u8视频合并技术完整实现方案
大数据·前端·数据库
尽兴-4 小时前
4.1 智能体核心:Agent、Sub-Agent、ReAct、规划执行
前端·javascript·react.js·agent·react·subagent