JS如何实现深拷贝_JavaScript深拷贝与浅拷贝区别及实现方法

深拷贝与浅拷贝的核心区别在于对引用类型处理方式不同:浅拷贝仅复制对象第一层属性,嵌套对象仍共享引用,修改会相互影响;深拷贝则递归复制所有层级,生成完全独立的新对象。常见浅拷贝方法包括Object.assign、扩展运算符、slice/concat等,但均只复制引用地址;JSON.parse(JSON.stringify(obj))可实现深拷贝,但无法处理函数、undefined、Symbol、循环引用及Date类型;手写递归深拷贝通过WeakMap解决循环引用问题,支持Date、RegExp等特殊类型,兼容性好;structuredClone为现代浏览器提供的原生深拷贝API,功能强大且支持复杂类型和循环引用,但兼容性有限。实际应用中需根据数据结构复杂度、特殊类型存在与否及浏览器兼容需求选择合适方案,手写递归通常是最稳妥选择。

js如何实现深拷贝_javascript深拷贝与浅拷贝区别及实现方法

JavaScript中的深拷贝和浅拷贝主要区别在于对引用类型数据的处理方式。当你复制一个对象时,如果该对象包含嵌套的对象或数组,浅拷贝只会复制第一层属性,而更深层的引用仍然指向原对象;深拷贝则会递归复制所有层级,生成一个完全独立的新对象。

浅拷贝 vs 深拷贝的区别

浅拷贝:只复制对象的第一层属性。对于基本类型值,复制的是值本身;对于引用类型(如对象、数组),复制的是引用地址,因此新旧对象共享嵌套结构,修改会影响彼此。

深拷贝:递归复制对象的所有层级,包括嵌套的对象和数组,生成一个全新的、与原对象无任何关联的独立副本。修改新对象不会影响原对象。

常见浅拷贝方法:Object.assign({}, obj) {…obj} Array.prototype.slice() / concat()注意:这些方法都只能实现浅拷贝。

JSON.stringify + JSON.parse 实现深拷贝

这是最简单的深拷贝方法之一:

立即学习“Java免费学习笔记(深入)”;

const deepCopy = JSON.parse(JSON.stringify(originalObj));

优点:语法简单,能处理大多数普通对象和数组的深拷贝。

缺点:有明显局限性:

无法处理函数、undefined、Symbol等非可序列化类型 会忽略 Symbol 属性键 不能处理循环引用(会报错) Date 对象会被转为字符串

因此,仅适用于纯数据对象(POJO)且不含特殊类型的场景。

手写递归实现深拷贝

为了克服 JSON 方法的限制,可以手动编写递归函数实现更完整的深拷贝:

function deepClone(obj, hash = new WeakMap()) {
if (obj == null) return obj;
if (typeof obj !== 'object') return obj;
if (obj instanceof Date) return new Date(obj);
if (obj instanceof RegExp) return new RegExp(obj);
// 防止循环引用
if (hash.has(obj)) return hash.get(obj);
// 保留构造函数信息
const clone = new obj.constructor();
hash.set(obj, clone);
for (let key in obj) {
if (obj.hasOwnProperty(key)) {
clone[key] = deepClone(obj[key], hash);
}
}
return clone;
}

这个版本支持 Date、RegExp,并使用 WeakMap 解决循环引用问题,兼容性较好。

使用 structuredClone(现代浏览器)

现代浏览器提供了原生的深拷贝 API:

const deepCopy = structuredClone(originalObj);

它支持很多复杂类型,包括 transferable objects,也支持循环引用。但兼容性有限(Chrome 98+ 等较新版本支持),不适合需要兼容老浏览器的项目。

基本上就这些常见的实现方式。选择哪种方法取决于你的具体需求:数据结构是否复杂、是否包含特殊类型、是否需要兼容老浏览器等。多数情况下,手写递归加 WeakMap 是比较稳妥的选择。

以上就是JS如何实现深拷贝_JavaScript深拷贝与浅拷贝区别及实现方法的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1536551.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月21日 03:22:52
下一篇 2025年12月21日 03:23:00

相关推荐

  • 动态网格重绘中的DOM管理与优化实践

    本文旨在解决动态调整css grid布局时,旧网格元素未清除导致布局错乱的问题。通过分析dom操作不当的根本原因,提供了一种在重新生成网格前清空容器内容的有效方法,并优化了事件监听机制,确保etch-a-sketch等交互式应用在尺寸调整时能正确、高效地渲染新网格,从而避免元素堆叠和性能下降。 1.…

    2025年12月21日
    000
  • jQuery中过滤方法slice()方法如何使用?

    jQuery的slice()方法用于筛选指定范围的元素,语法为$(selector).slice(start, end);start为起始索引(从0开始,负数表示从末尾计),end为结束位置(不包含,可选);常用于操作部分DOM元素,如前几项、中间段或最后几项;例如选取前3个div:$(&#8216…

    2025年12月21日
    000
  • 使用CARTO v3和DeckGL动态控制地图图层可见性教程

    本教程详细介绍了如何利用carto v3和deckgl库在javascript中实现地图图层的动态显示与隐藏。核心机制在于通过修改`deckgl`实例的`layers`属性,特别是每个图层对象的`visible`属性,并结合`deckgl.setprops()`方法来实时更新地图视图。文章涵盖了环境…

    2025年12月21日
    000
  • FullCalendar多实例同步:主日历更新后列表日历自动刷新指南

    本文详细介绍了在使用FullCalendar时,如何解决在一个页面中管理多个日历实例并实现数据同步的问题。当主编辑日历更新事件后,旁边的列表视图日历能够自动刷新其事件数据。核心解决方案在于将日历对象声明为全局变量,并在主日历的AJAX数据更新成功回调中,调用列表日历的`refetchEvents()…

    2025年12月21日
    000
  • 使用JavaScript实现一个简单的Web服务器_Node.js

    使用Node.js内置http模块可快速创建Web服务器,通过createServer处理请求响应,监听端口并根据URL实现简单路由,返回文本或HTML内容。 使用Node.js实现一个简单的Web服务器 Node.js 是基于 Chrome V8 引擎的 JavaScript 运行环境,它让 Ja…

    2025年12月21日
    000
  • 使用WebRTC构建点对点视频聊天应用

    答案是掌握WebRTC的信令机制、连接流程和网络穿透。通过SDP协商媒体能力,ICE候选建立点对点连接,利用WebSocket实现信令服务器交换offer/answer,结合STUN/TURN解决NAT穿透,前端处理媒体获取与连接状态,确保低延迟实时通信。 点对点视频聊天应用的核心在于实时音视频传输…

    2025年12月21日
    000
  • JavaScript 数据类型:深入理解原始类型与引用类型

    JavaScript数据类型分原始类型和引用类型,原始类型按值访问、不可变,包括string、number、boolean、null、undefined、symbol、bigint;2. 原始类型赋值时复制值副本,互不影响;3. 引用类型存储对象地址,赋值时复制引用,指向同一对象则修改互相影响;4.…

    2025年12月21日
    000
  • JavaScript数组去重的十种高效方法_javascript技巧

    使用Set去重最简洁高效,适用于基本类型;2. filter+indexOf兼容性好但性能较差;3. forEach+Object利用键值记录,需注意类型转换问题;4. Map方式更安全可靠,支持任意键类型;5. reduce函数式风格清晰但性能一般;6. 排序后比较节省空间但改变顺序;7. JSO…

    2025年12月21日
    000
  • JavaScript与Spring多数据源配置结合的方法

    JavaScript无法直接配置Spring多数据源,但可通过调用不同API触发后端数据源切换。后端通过@DataSource注解和AbstractRoutingDataSource实现动态路由,前端根据业务需求请求对应接口,如读操作调用/report走从库,写操作调用/user走主库。配合CORS…

    2025年12月21日
    000
  • jquery中在元素之前添加元素的before() 方法

    before() 方法用于在被选元素前插入内容,作为兄弟节点出现;2. 可插入 HTML 字符串、DOM 元素、jQuery 对象等;3. 与 prepend()(内部插入)和 after()(后置插入)不同,before() 在外部前置内容。 在 jQuery 中,before() 方法用于在被选…

    2025年12月21日
    000
  • Supabase Edge Function CORS 故障排除指南

    本文旨在解决在 Supabase Edge Function 中遇到的跨域资源共享(CORS)错误,特别是当函数在本地正常运行但在生产环境失败时。文章将详细阐述如何在 Edge Function 中正确配置 CORS 头部,处理预检请求,并强调本地 Supabase Docker 环境与 CLI 同…

    2025年12月21日
    000
  • dc.js barChart 分组与维度:自定义分箱与刷选机制深度解析

    本文深入探讨dc.js中`dc.barChart`的维度(dimension)和分组(group)机制,特别是如何实现自定义数据分箱。我们将对比在维度函数内或分组函数内进行分箱的两种方法,并通过具体代码示例展示其实现。文章还将重点解析刷选(brushing)功能对这两种分箱策略的影响,强调在交互式数…

    2025年12月21日
    000
  • AJAX登录表单成功提交后页面不跳转的解决方案

    本教程旨在解决使用javascript和ajax处理登录表单时,即便数据验证成功,表单也未能按预期跳转的问题。我们将深入分析`event.preventdefault()`和`form.submit()`在ajax成功回调中的常见误区,并提供一套正确的客户端重定向策略,确保用户在成功登录后能够无缝导…

    2025年12月21日
    000
  • Phaser中根据速度方向动态调整精灵旋转角度

    本教程详细介绍了如何在Phaser框架中实现精灵(Sprite)根据其当前运动速度方向进行动态旋转。文章将涵盖精灵初始化时的方向设定、处理与世界边界碰撞后的方向更新,以及处理精灵之间碰撞后的方向调整。通过结合Phaser的物理系统、事件监听和向量数学,读者将学会如何确保精灵始终面向其前进方向,从而提…

    2025年12月21日
    000
  • JS注解怎么标注异步函数_ JS异步函数注解的书写与使用说明

    异步函数的注解需用JSDoc标注Promise返回类型,如@returns {Promise},并可用@async标识函数为异步,配合@param描述参数,提升代码可读性与IDE提示能力。 在JavaScript中,并没有像Java那样的“注解”(Annotation)语法,因此所谓的“JS注解”通…

    2025年12月21日
    000
  • dom节点删除方法detach()和remove()有什么区别

    remove()彻底删除元素及其数据和事件,适用于永久移除;2. detach()保留数据和事件,适用于临时移除后重新插入。 在 jQuery 中,detach() 和 remove() 都用于从 DOM 中删除元素,但它们的关键区别在于是否保留元素的 jQuery 数据和绑定事件。 1. remo…

    2025年12月21日
    000
  • WebGL鼠标事件绘制像素:理解顶点属性与绘制调用

    本教程详细介绍了如何在webgl画布上通过鼠标事件绘制单个像素。文章深入探讨了`vertexattrib2f`与`vertexattribpointer`的区别及适用场景,纠正了常见的`drawarrays`调用错误和缓冲区管理误区,并提供了完整的代码示例,帮助开发者理解webgl中javascri…

    2025年12月21日
    000
  • JavaScript实现一个简单的MVVM框架_js框架原理

    答案:通过Object.defineProperty实现数据劫持,结合模板编译与Watcher订阅者模式,构建极简MVVM框架,实现数据变化自动更新视图的核心机制。 要实现一个简单的MVVM(Model-View-ViewModel)框架,核心是数据绑定和响应式更新。我们可以通过JavaScript…

    2025年12月21日
    000
  • dc.js 条形图分组与维度:深度解析自定义分箱策略及其对刷选功能的影响

    `dc.js` 条形图在数据分箱时,存在于 `group()` 或 `dimension()` 函数中定义分箱逻辑的两种策略。本文详细阐述如何在 `group()` 中实现自定义区间分箱,并揭示该方法在支持交互式刷选(brushing)功能时的局限性。通过对比分析,我们将推荐一种更适用于需要刷选功能…

    2025年12月21日
    000
  • 使用 CARTO v3 和 DeckGL 实现交互式地图层管理与动态显示

    本教程详细介绍了如何利用 CARTO v3 和 DeckGL 库构建交互式地图应用,重点讲解了动态显示/隐藏地图层、实现悬停工具提示以及管理图层状态的核心技术。通过 deckgl.setProps() 方法,结合 visible 属性和事件监听,开发者可以高效地控制地图元素的可见性,并提升用户体验。…

    2025年12月21日
    000

发表回复

登录后才能评论
关注微信