JavaScript数据深浅拷贝的深度解析(一文明白!)

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: JavaScript数据深浅拷贝的深度解析(一文明白!)

涉及到知识点:JavaScript数据类型,JavaScript的数据存储方式(堆栈);

首先我们要理解什么是深浅拷贝?深浅拷贝是JavaScript对数据的操作,但是深浅拷贝是相对于JavaScript的数据类型来说的,在JavaScript中基本数据类型是没有深浅拷贝之分的,深浅拷贝只是说对于JavaScript的复杂数据类型(引用数据类型来说的);基本数据类型都是存储在栈区的,而深浅拷贝也是操作的栈区(只不过基本数据类型操作的值,而引用数据类型操作的是内存地址),所以基本数据类型不存在深浅拷贝之分;

引用数据类型的深浅拷贝是什么? 引用数据类型的深浅拷贝区别在于,浅拷贝是把存储在栈区里面的堆区内存地址直接复制给一个变量(因此只要指向这个地址的变量里面的数据都是联动的),而深拷贝是把当前存储在栈区的堆区内存地址进行访问且在堆区在划分一块区域进行存储访问过来的数据; 理解了深浅拷贝的概念,接下来采用代码进行实现

//经过测试的代码
//对基础数据类型进行拷贝,基本数据类型不存在深浅拷贝之分
//基本数据类型有:String,Number,Boolean,null,undefined,Symbol
//引用数据类型:Object,Array, Function
var a=1;
var b=a;
a=3;
console.log(a,b) //3,1

引用数据类型的浅拷贝

//浅拷贝
var a={};
var b=a;
a.name='王菜菜';
console.log(a,b) //{ name: "王菜菜" },{ name: "王菜菜" }
//Object.assign(a)第一个参数不传递空对象的话就是浅拷贝,传了就是深拷贝,将会在下面讲到
var a={
name:1,
arr:[1,2,3,4]
};
var b=Object.assign(a);
a.name='王菜菜';
a.arr='之前是数组';
console.log(a,b);
//{name: "王菜菜", arr: "之前是数组"} {name: "王菜菜", arr: "之前是数组"}
//通过for in 方法进行深拷贝
var a={
name:'前端',
arr:{age:22}
};
function CopyObj(obj) {
    // 判断是不是数组
   let _obj = Array.isArray(obj) ? [] : {};
   for (let i in obj) {
   _obj[i] = obj[i];
  }
   return obj;
}
var b=CopyObj(a);
a.name='王菜菜';
a.arr.age='年龄';
console.log(a,b);
//{name: "王菜菜",arr: {age: "年龄"}},{arr: {name: "前端",age: "年龄"}}

引用数据的深拷贝

//对象赋值的方式实现深拷贝
var a={
name:'前端'
};
var b={
name:a.name
};
a.name='王菜菜';
console.log(a,b)
//通过json的方法进行深拷贝,但是无法实现对象中方法的深拷贝
//原因:undefined,function,symbol 会在转换json的过程中被忽略掉;
var a={
name:'前端',
fn:function(){
console.log('2021加油!!!');
}
};
function CopyObj(obj) {
  let _obj = JSON.stringify(obj);
  return JSON.parse(_obj);
}
 let b = CopyObj(a);
a.name='王菜菜';
console.log('当前数据',d,a); //{ name: "前端" },{ name: "王菜菜", fn: ƒ}
// 网上找的:Object.assign()方法实现深拷贝,但是也划分深浅拷贝,对象只有一级时为深拷贝,
//二级之后就是浅拷贝;
var a={
name:'前端',
arr:{age:22}
};
var b=Object.assign({},a);
a.name='王菜菜';
a.arr.age='年龄';
console.log(a,b);
//数据联动是浅拷贝,数据不联动是深拷贝
//{name: "王菜菜", arr: "{age:'年龄'}"} {name: "前端", arr: "{age:'年龄'}"}
var a={
name:'前端',
arr:{age:22}
};
var b={...a}; //这样只能拷贝一第一次,但是...扩展运算符可以进行展开继续拷贝;
var b={...a,arr:{...a.arr}};//深拷贝,只拷贝2层的深拷贝;
a.name='王菜菜';
a.arr.age='年龄';
console.log(a,b);

对数组实现深拷贝

//这个是es6的新语法,用于将两类对象转换成真正的数组:类似数组的对象 和 可遍历对象;
var a = ['前端', ['王菜菜', '2021'], '加油!'];
//需要伪数组对象或可迭代对象;
//传递可迭代的对象就需要传递第二个参数,是个回调函数,然后把对象处理一下
var b = Array.from(a);
a=[];
console.log(a,b); 
//[], Array:['前端',Array: ['王菜菜', '2021'], '加油!']
//es6的新语法,不会改变现有的数组,而仅仅会返回被连接后的数组;
var a = ['前端', ['王菜菜', '2021'], '加油!'];
var b = a.concat([1,2]);
a=[];
console.log(a,b); 
// Array [] Array ["前端", Array ["王菜菜", "2021"], "加油!", 1, 2]
var a = ['前端', ['王菜菜', '2021'], '加油!'];
//截取,第一个参数从下标为几开始截取,第二个参数在下标为几结束截取,结束的下标数据不截取;
var b = a.slice(0,3);
a=[];
console.log(a,b);
//Array [] Array ["前端", Array ["王菜菜", "2021"], "加油!"]

上面的深拷贝都是根据JavaScript提供的自带api实现的,下面将采用不使用api进行实现

//采用递归进行实现(不过我这个方式还是有些不严谨)
function CopyObj(obj) {
//判断是数组还是对象
  let _obj = Array.isArray(obj) ? [] : {};
  //进行对象循环
  for (let key in obj) {
    //判断obj对象中是否含有当前这个key值,有则为true,否则为false
    if (obj.hasOwnProperty(key)) {
    //obj对象中的key值存在且key值为第二层对象
      if (obj[key] && typeof obj[key] === 'object') {
      //进行函数递归
        _obj[key] = CopyObj(obj[key]);
      } else {
      //如果不是对象则直接赋值
        _obj[key] = obj[key];
      }
    }
  }
  //递归的出口
  return _obj;
}

想一起学习前端技术进行讨论的可以私信我!!!

相关文章
|
2月前
|
JavaScript 前端开发 Go
CSS 与 JS 对 DOM 解析和渲染的影响
【10月更文挑战第16天】CSS 和 JS 会在一定程度上影响 DOM 解析和渲染,了解它们之间的相互作用以及采取适当的优化措施是非常重要的。通过合理的布局和加载策略,可以提高网页的性能和用户体验,确保页面能够快速、流畅地呈现给用户。在实际开发中,要根据具体情况进行权衡和调整,以达到最佳的效果。
|
2月前
|
存储 前端开发 JavaScript
JavaScript垃圾回收机制深度解析
【10月更文挑战第21】JavaScript垃圾回收机制深度解析
110 59
|
2月前
|
自然语言处理 数据可视化 前端开发
从数据提取到管理:合合信息的智能文档处理全方位解析【合合信息智能文档处理百宝箱】
合合信息的智能文档处理“百宝箱”涵盖文档解析、向量化模型、测评工具等,解决了复杂文档解析、大模型问答幻觉、文档解析效果评估、知识库搭建、多语言文档翻译等问题。通过可视化解析工具 TextIn ParseX、向量化模型 acge-embedding 和文档解析测评工具 markdown_tester,百宝箱提升了文档处理的效率和精确度,适用于多种文档格式和语言环境,助力企业实现高效的信息管理和业务支持。
4030 5
从数据提取到管理:合合信息的智能文档处理全方位解析【合合信息智能文档处理百宝箱】
|
1月前
|
存储 分布式计算 Java
存算分离与计算向数据移动:深度解析与Java实现
【11月更文挑战第10天】随着大数据时代的到来,数据量的激增给传统的数据处理架构带来了巨大的挑战。传统的“存算一体”架构,即计算资源与存储资源紧密耦合,在处理海量数据时逐渐显露出其局限性。为了应对这些挑战,存算分离(Disaggregated Storage and Compute Architecture)和计算向数据移动(Compute Moves to Data)两种架构应运而生,成为大数据处理领域的热门技术。
48 2
|
1月前
|
数据采集 存储 JavaScript
如何使用Puppeteer和Node.js爬取大学招生数据:入门指南
本文介绍了如何使用Puppeteer和Node.js爬取大学招生数据,并通过代理IP提升爬取的稳定性和效率。Puppeteer作为一个强大的Node.js库,能够模拟真实浏览器访问,支持JavaScript渲染,适合复杂的爬取任务。文章详细讲解了安装Puppeteer、配置代理IP、实现爬虫代码的步骤,并提供了代码示例。此外,还给出了注意事项和优化建议,帮助读者高效地抓取和分析招生数据。
如何使用Puppeteer和Node.js爬取大学招生数据:入门指南
|
1月前
|
JavaScript API 开发工具
<大厂实战场景> ~ Flutter&鸿蒙next 解析后端返回的 HTML 数据详解
本文介绍了如何在 Flutter 中解析后端返回的 HTML 数据。首先解释了 HTML 解析的概念,然后详细介绍了使用 `http` 和 `html` 库的步骤,包括添加依赖、获取 HTML 数据、解析 HTML 内容和在 Flutter UI 中显示解析结果。通过具体的代码示例,展示了如何从 URL 获取 HTML 并提取特定信息,如链接列表。希望本文能帮助你在 Flutter 应用中更好地处理 HTML 数据。
109 1
|
20天前
|
数据采集 存储 自然语言处理
基于Qwen2.5的大规模ESG数据解析与趋势分析多Agent系统设计
2022年中国上市企业ESG报告数据集,涵盖制造、能源、金融、科技等行业,通过Qwen2.5大模型实现报告自动收集、解析、清洗及可视化生成,支持单/多Agent场景,大幅提升ESG数据分析效率与自动化水平。
|
1月前
|
前端开发 JavaScript
JavaScript新纪元:ES6+特性深度解析与实战应用
【10月更文挑战第29天】本文深入解析ES6+的核心特性,包括箭头函数、模板字符串、解构赋值、Promise、模块化和类等,结合实战应用,展示如何利用这些新特性编写更加高效和优雅的代码。
43 0
|
1月前
|
JSON 前端开发 JavaScript
API接口商品详情接口数据解析
商品详情接口通常用于提供特定商品的详细信息,这些信息比商品列表接口中的信息更加详细和全面。以下是一个示例的JSON数据格式,用于表示一个商品详情API接口的响应。这个示例假定API返回一个包含商品详细信息的对象。
|
23天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
61 2

推荐镜像

更多