如何深入理解 JavaScript 引用类型的浅拷贝和深拷贝

首页 / 常见问题 / 低代码开发 / 如何深入理解 JavaScript 引用类型的浅拷贝和深拷贝
作者:代码开发工具 发布时间:12-28 19:29 浏览量:4030
logo
织信企业级低代码开发平台
提供表单、流程、仪表盘、API等功能,非IT用户可通过设计表单来收集数据,设计流程来进行业务协作,使用仪表盘来进行数据分析与展示,IT用户可通过API集成第三方系统平台数据。
免费试用

深入理解JavaScript引用类型的浅拷贝和深拷贝,首先要明确它们在复制对象时对内存中数据的处理方式不同浅拷贝(Shallow Copy)仅复制对象的第一层属性,如果属性值是引用类型,则复制的是内存地址。在浅拷贝后,原始对象和新对象会共享同一块内存地址中存储的引用值。相比之下,深拷贝(Deep Copy)则是完全复制一个新的对象,不仅复制对象的第一层属性,还包括内嵌的对象和数组。深拷贝后,新对象和原始对象完全独立,修改新对象不会影响原始对象的内部属性。

对于浅拷贝而言,因为它复制的是内存中的地址,所以两个对象的引用类型属性实际指向同一块内存空间。当我们在一个对象上修改这类属性时,另一个对象也会受到影响,这在一些场合下可能引发问题。如需避免这种连带影响,我们就需要使用深拷贝。

一、浅拷贝的实现

浅拷贝在JavaScript中可以通过多种方法实现,例如使用Object.assign方法或是展开运算符(…)。

Object.assign 方法

Object.assign 方法可以将所有可枚举属性的值从一个或多个源对象复制到目标对象,并返回这个目标对象。使用它进行浅拷贝时,它仅复制对象本身的属性,并不会复制对象内部嵌套的对象。

let original = { a: 1, b: { c: 2 } };

let copy = Object.assign({}, original);

在这段代码中,original 对象的属性 b 是一个嵌套对象。通过Object.assign 实现浅拷贝后,copy 对象的 b 属性与 original 对象的 b 属性实际上指向同一块内存地址。

展开运算符

展开运算符(spread operator)... 也可以用来创建一个对象的浅拷贝。这种方法类似于 Object.assign,但更为简洁。

let original = { a: 1, b: { c: 2 } };

let copy = { ...original };

使用展开运算符,新创建的 copy 对象包含了 original 对象所有层级的浅复制,内嵌的对象仍然是共享引用。

二、深拷贝的实现

深拷贝意味着复制对象所有层级的属性,不共享任何引用。在JavaScript中,实现深拷贝可以使用JSON方法或库函数。

JSON 方法

最简单的深拷贝实现方式是利用 JSON.stringifyJSON.parse。首先将对象转换成JSON字符串,然后再将这个字符串解析回对象。

let original = { a: 1, b: { c: 2 } };

let copy = JSON.parse(JSON.stringify(original));

这种方法的局限性在于它无法复制函数和正则对象等不能转换为JSON的值。因此,只有当对象只包含数字、字符串、数组和普通对象时,这种方法才有效。

使用库

JavaScript社区提供了许多用于深拷贝的库函数,如lodash的 _.cloneDeep 方法。

let original = { a: 1, b: { c: 2 } };

let copy = _.cloneDeep(original);

lodash的 cloneDeep 方法能够深拷贝对象,包括那些 JSON.stringifyJSON.parse 无法处理的类型,如函数、正则表达式、Maps和Sets等。

三、浅拷贝与深拷贝的选择

选择浅拷贝还是深拷贝,主要取决于应用场景及对原始数据的操作需求。

使用场景

浅拷贝通常在不涉及多层嵌套对象的情况下使用,或者当我们需要复制一个对象,但又要保留与原对象相同的引用时。例如,在React或Vue等框架中,有时我们会通过浅拷贝来触发组件的重新渲染,因为引用变了,但实际数据并没有改变。

深拷贝适用于需要完全独立复制的情况,特别是当对象包含多层嵌套属性,且每层都需要独立修改时。在一些重要的数据处理或状态管理(如使用Redux进行状态管理)中,深拷贝帮助我们避免不必要的副作用。

性能考量

由于深拷贝需要复制对象中的所有元素,所以它比浅拷贝消耗更多的资源和时间。在性能敏感的应用中,过度使用深拷贝可能会导致内存和运行时间上的消耗。因此在不需要深层独立操作的情况下,应该尽量使用浅拷贝。

四、浅拷贝和深拷贝的注意事项

在使用浅拷贝和深拷贝时,有几个注意事项需要考虑。

循环引用问题

当对象自我引用,或者两个对象相互引用形成循环时,深拷贝特别是通过 JSON.stringifyJSON.parse 的方法可能会引发错误。递归的深拷贝实现方法需要检测并处理循环引用的情况。

特殊对象和类型

在深拷贝过程中,需要考虑如何处理特殊对象和类型(如Date对象、正则对象等),因为 JSON.stringify 方法无法正确处理它们。为这些类型实现深拷贝可能需要特定的策略或使用第三方库。

结论

JavaScript中的浅拷贝和深拷贝是处理对象复制的两种重要技术,它们具有不同的使用场景和性能影响。理解它们的差异对于编写高效且错误较少的代码非常关键。在实际开发过程中,要根据具体需求和对象的结构特点选择合适的拷贝方法,并注意避免循环引用等可能导致问题的情况。

相关问答FAQs:

什么是 JavaScript 引用类型的浅拷贝和深拷贝?

JavaScript 中,引用类型的拷贝方式有两种:浅拷贝和深拷贝。浅拷贝只复制了对象的引用,而深拷贝会复制对象的所有值和属性。

如何进行 JavaScript 引用类型的浅拷贝?

要进行 JavaScript 引用类型的浅拷贝,可以使用 Object.assign() 方法或者展开运算符(…)来复制对象。这种方式只复制了对象的引用,当修改拷贝后的对象时,原始对象也会受到影响。

如何进行 JavaScript 引用类型的深拷贝?

如果需要进行深拷贝,可以使用 JSON.parse(JSON.stringify(obj)) 方法。这种方式会将对象转换为 JSON 字符串,然后再将 JSON 字符串转换回对象,从而实现深拷贝。需要注意的是,使用该方法进行深拷贝时,会忽略对象的函数、正则表达式等特殊属性。如果对象中包含这些特殊属性,可能会造成深拷贝的失效。

如何判断 JavaScript 引用类型的拷贝方式是浅拷贝还是深拷贝?

要确定 JavaScript 引用类型的拷贝方式是浅拷贝还是深拷贝,可以通过修改拷贝后的对象来观察原始对象是否受到影响。如果修改拷贝后的对象不会影响原始对象,那么是深拷贝。如果修改拷贝后的对象也会同时修改原始对象,那么是浅拷贝。可以使用 Object.is() 方法来比较两个对象是否相等,进一步验证拷贝方式。

最后建议,企业在引入信息化系统初期,切记要合理有效地运用好工具,这样一来不仅可以让公司业务高效地运行,还能最大程度保证团队目标的达成。同时还能大幅缩短系统开发和部署的时间成本。特别是有特定需求功能需要定制化的企业,可以采用我们公司自研的企业级低代码平台织信Informat。 织信平台基于数据模型优先的设计理念,提供大量标准化的组件,内置AI助手、组件设计器、自动化(图形化编程)、脚本、工作流引擎(BPMN2.0)、自定义API、表单设计器、权限、仪表盘等功能,能帮助企业构建高度复杂核心的数字化系统。如ERP、MES、CRM、PLM、SCM、WMS、项目管理、流程管理等多个应用场景,全面助力企业落地国产化/信息化/数字化转型战略目标。 版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们微信:Informat_5 处理,核实后本网站将在24小时内删除。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系邮箱:hopper@cornerstone365.cn 处理,核实后本网站将在24小时内删除。

最近更新

为什么要把 JavaScript 放到服务器端上运行
12-28 19:29
什么编程语言能和Java搭配在一起做web开发
12-28 19:29
对于JAVA有没有更适合初学者的开发词典
12-28 19:29
如何更加通俗易懂 JavaScript
12-28 19:29
为何许多人不建议从 Javascript 入门学习计算机
12-28 19:29
urldecode 如何用 JavaScript 实现
12-28 19:29
开发一个网站,只用css、HTML、JavaScript够用吗
12-28 19:29
java的框架都有哪些
12-28 19:29
能用 VBScript 做出类似 JavaScript 那种动态效果么
12-28 19:29

立即开启你的数字化管理

用心为每一位用户提供专业的数字化解决方案及业务咨询

  • 深圳市基石协作科技有限公司
  • 地址:深圳市南山区科技中一路大族激光科技中心909室
  • 座机:400-185-5850
  • 手机:137-1379-6908
  • 邮箱:sales@cornerstone365.cn
  • 微信公众号二维码

© copyright 2019-2024. 织信INFORMAT 深圳市基石协作科技有限公司 版权所有 | 粤ICP备15078182号

前往Gitee仓库
微信公众号二维码
咨询织信数字化顾问获取最新资料
数字化咨询热线
400-185-5850
申请预约演示
立即与行业专家交流