实现 js 中所有对象的深拷贝(包装对象,Date 对象,正则对象)

news2025/1/10 11:35:40

通过递归可以简单实现对象的深拷贝,但是这种方法不管是 ES6 还是 ES5 实现,都有同样的缺陷,就是只能实现特定的 object 的深度复制(比如数组和函数),不能实现包装对象 Number,String , Boolean,以及 Date 对象,RegExp 对象的复制。

(1)简单的深拷贝

function deepClone(obj) {
  var newObj = obj instanceof Array ? [] : {}
 
  for (var i in obj) {
    newObj[i] = typeof obj[i] == 'object' ? deepClone(obj[i]) : obj[i]
  }
 
  return newObj
}


这种方法可以实现一般对象和数组对象的拷贝,比如:

var arr = [1, 2, 3]
 
var newArr = deepClone(arr)
 
// newArr->[1,2,3]
 
var obj = {
  x: 1,
  y: 2
}
 
var newObj = deepClone(obj)
 
// newObj={x:1,y:2}


但是不能实现例如包装对象 Number,String,Boolean,以及正则对象 RegExp 和 Date 对象的拷贝,比如:

//Number 包装对象
 
var num = new Number(1)
 
typeof num // "object"
 
var newNum = deepClone(num)
 
//newNum -> {} 空对象
 
//String 包装对象
 
var str = new String('hello')
 
typeof str //"object"
 
var newStr = deepClone(str)
 
//newStr-> {0:'h',1:'e',2:'l',3:'l',4:'o'};
 
//Boolean 包装对象
 
var bol = new Boolean(true)
 
typeof bol //"object"
 
var newBol = deepClone(bol)
 
// newBol ->{} 空对象


(2)valueof()函数

所有对象都有 valueOf 方法,valueOf 方法对于:如果存在任意原始值,它就默认将对象转换为表示它的原始值。对象是复合值,而且大多数对象无法真正表示为一个原始值, 因此默认的 valueOf()方法简单地返回对象本身,而不是返回一个原始值。数组、函数和正则表达式简单地继承了这个默认方法,调用这些类型的实例的 valueOf()方法只是简单返回这个对象本身。

对于原始值或者包装类:

function baseClone(base) {
  return base.valueOf()
}
 
//Number
 
var num = new Number(1)
 
var newNum = baseClone(num)
 
//newNum->1
 
//String
 
var str = new String('hello')
 
var newStr = baseClone(str)
 
// newStr->"hello"
 
//Boolean
 
var bol = new Boolean(true)
 
var newBol = baseClone(bol)
 
//newBol-> true


其实对于包装类,完全可以用=号来进行拷贝,其实没有深拷贝一说,这里用 valueOf 实现,语法上比较符合规范。

对于 Date 类型:

因为 valueOf 方法,日期类定义的 valueOf()方法会返回它的一个内部表示:1970 年 1 月 1 日以来的毫秒数.因此我们可以在 Date 的原型上定义拷贝的方法:

Date.prototype.clone = function () {
  return new Date(this.valueOf())
}
 
var date = new Date('2010')
 
var newDate = date.clone()
 
// newDate-> Fri Jan 01 2010 08:00:00 GMT+0800
对于正则对象 RegExp:

RegExp.prototype.clone = function () {
  var pattern = this.valueOf()
 
  var flags = ''
 
  flags += pattern.global ? 'g' : ''
 
  flags += pattern.ignoreCase ? 'i' : ''
 
  flags += pattern.multiline ? 'm' : ''
 
  return new RegExp(pattern.source, flags)
}
 
var reg = new RegExp('/111/')
 
var newReg = reg.clone()
 
//newReg-> /\/111\//


最终解决方案:

// 在深拷贝的基础上,对(包装对象Number,String,Boolean;Date 对象,RegExp正则对象)进行深拷贝

function deepClone(obj) {
  let newObj = obj instanceof Array ? [] : {}
  for (let i in obj) {
    // for...in 会遍历原型上的属性,此处只拷贝obj对象自身的属性
    if (obj.hasOwnProperty(i)) {
      let type = Object.prototype.toString.call(obj[i])
      if (typeof obj[i] == 'object') {
        // 拷贝的值为对象,则需要深拷贝
        if (type == '[object Date]') {
          newObj[i] = new Date(obj[i].valueOf())
        } else if (type == '[object RegExp]') {
          // 正则对象
          let pattern = obj[i].valueOf()
          let flags = ''
          flags += pattern.global ? 'g' : ''
          flags += pattern.ignoreCase ? 'i' : ''
          flags += pattern.multiline ? 'm' : ''
          newObj[i] = new RegExp(pattern.source, flags)
        } else if (type == '[object Array]' || type == '[object Object]') {
          // 数组或对象
          newObj[i] = deepClone(obj[i])
        } else {
          // 包装对象Number,String,Boolean
          newObj[i] = obj[i].valueOf()
        }
      } else if (typeof obj[i] == 'function') {
        // 函数
        newObj[i] = new Function('return ' + obj[i].toString())()
      } else {
        // 拷贝的值为原始值,则直接复制该值
        newObj[i] = obj[i]
      }
    }
  }
  return newObj
}


测试:


let obj = {
  name: 'zs',
  age: 20,
  food: ['apple', 'banana', 'orange'],
  obj1: {
    school: 'nyist'
  },
  reg: new RegExp('/A-Z/', 'gi'),
  date: new Date(),
  bol: new Boolean(true),
  num: new Number(999),
  fn: function () {
    this.age++
  }
}
 
let res = deepClone(obj)
 
res.obj1.school = '清华'
res.fn = function (a, b) {
  console.log(a, b)
}
res.food[1] = '香蕉'
 
console.log('res', res, 'obj', obj)
console.log(res.reg === obj.reg)

原文链接:https://blog.csdn.net/weixin_52624519/article/details/129265211

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/987187.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

如何压缩图片大小?缩小图片体积跟我学

在日常生活中,我们常常需要处理图片,但是由于图片大小过大,常常带来许多不便。那么,如何压缩图片大小呢?下面就为大家介绍三个方法,让你轻松解决这个问题。 一、使用图片编辑软件 市面上有许多图片编辑软件…

使用HTTP代理上网安全吗?

HTTP代理是一种代理服务器,它可以充当客户端和服务器之间的中介,以帮助客户端访问服务器上的资源。虽然使用HTTP代理可以带来一些便利,但是在安全方面也存在一些问题。 HTTP代理的安全问题 窃取用户信息 如果HTTP代理服务器不受信任&#xff…

【计算机网络】https协议

目录 概念的准备 什么是加密 为什么需要加密 常见的加密方式 对称加密 非对称加密 数据摘要(数字指纹) 数字签名 https的工作过程 方案一:只使用对称加密 方案二:只使用非对称加密 方案三:双方都采用非对称加密 方案四&#xff…

【Spring Boot 源码学习】深入 FilteringSpringBootCondition

走近 AutoConfigurationImportFilter 引言往期内容主要内容1. match 方法2. ClassNameFilter 枚举类3. filter 方法 总结 引言 前两篇博文笔者带大家从源码深入了解了 Spring Boot 的自动装配流程,其中自动配置过滤的实现由于篇幅限制,还未深入分析。 …

2023国赛数学建模C题思路模型 - 蔬菜类商品的自动定价与补货决策

# 1 赛题 在生鲜商超中,一般蔬菜类商品的保鲜期都比较短,且品相随销售时间的增加而变差, 大部分品种如当日未售出,隔日就无法再售。因此, 商超通常会根据各商品的历史销售和需 求情况每天进行补货。 由于商超销售的蔬菜…

uniapp使用webview将页面转换成图片支持h5、app、小程序

uniapp使用webview将页面转换成图片支持h5、app、小程序 在uniapp项目中新建主页和webview页面 index.vue代码 <template><view><!-- 微信小程序要设置src为网络路径 --><web-view src"/hybrid/html/webview.html"></web-view><…

十四、内置模块path、邂逅Webpack和打包过程、css-loader

一、内置模块path &#xff08;1&#xff09;path介绍 &#xff08;2&#xff09; path常见的API 这里重点讲一下path.resolve()。 看上面的例子&#xff0c;从右往左开始解析&#xff0c;所以一开始解析的就是 /abc.txt &#xff0c;这个时候就会把它当成一个绝对路径了&am…

C#,数值计算——用于积分函数与方法的Stiel类的计算方法与源程序

1 文本格式 using System; namespace Legalsoft.Truffer { public class Stiel { public class pp : UniVarRealValueFun, RealValueFun { public Stiel st { get; set; } null; public pp() { } public doubl…

探索数据库管理的利器 - PHPMyAdmin

有一个项目&#xff0c;后端由博主独自负责&#xff0c;最近需要将项目交接给另一位同事。在项目初期&#xff0c;博主直接在数据库中使用工具创建了相关表格&#xff0c;并在完成后利用PhpMyAdmin生成了一份数据字典&#xff0c;供团队使用。然而&#xff0c;在随后的开发过程…

JDK1.8下载、安装和环境配置使用

JDK1.8下载、安装和配置 下载安装包解压文件配置测试安装 下载安装包 链接地址 https://pan.baidu.com/s/1RF7-ulq0_qAelpXskDxdvA 提取码 d1y0解压文件 jdk1.8.0_181 配置 右击我的电脑&#xff0c;选择属性 2.点击高级系统设置 在系统变量区里点击&#xff1a;新建…

听书网站模板源码 懒人书院网站源码 苹果cms手机听书网站模版源码 支持手机端

苹果cms超漂亮UI高仿芒果TV听书网站模板带手机端。 手机版修改logo&#xff0c;ting_wap/images/logo.png 电脑版修改logo&#xff0c;ting_pc/img/logo.png 编辑推荐后台推荐5颗星。 新势力/热播榜单后台推荐9颗星。

3D基础:Y-Up和Z-Up

推荐&#xff1a;用 NSDT编辑器快速搭建可编程3D场景 所有 3D 工具都包含具有 X、Y 和 Z 轴的 3 维环境。 这些工具中的 X 轴方向相同&#xff0c;即使用前视图时从左到右的水平线。 但是&#xff0c;不同的 3D 工具可能具有不同的 Y 轴和 Z 轴方向。 有些3D工具是Y-Up的&…

西门子S7-200 SMART软件的下载安装步骤

文章目录 1、软件下载2、软件安装 1、软件下载 访问西门子官网&#xff1a;https://www.siemens.com/cn/zh.html&#xff0c;进入后在左上角产品与服务&#xff0c;依次进入该目录&#xff1a; 实在找不到在右上角放大镜直接搜索smart200 点击进入即可&#xff0c;然后就进入…

浅谈安科瑞ADL200仪表在爱尔兰工厂的应用

摘要&#xff1a;用户端消耗着整个电网80%的电能&#xff0c;用户端智能化用电管理对用户可靠、安全、节约用电有十分重要的意义。构建智能用电服务体系&#xff0c;推广用户端智能多功能仪表、智能用电管理终端等设备用电管理解决方案&#xff0c;实现电网与用户的双向良性互动…

第 2 章 线性表 (线性表的静态单链表存储结构(一个数组可生成若干静态链表)实现)

1. 背景说明 静态单链表实现类似于单链表&#xff0c;只是指针域变成了数组下标。 2. 示例代码 1) status.h /* DataStructure 预定义常量和类型头文件 */#ifndef STATUS_H #define STATUS_H/* 函数结果状态码 */ #define TRUE 1 /* 返回值为真 */ #define FALSE …

从策略到执行:实施战略定位的实战手册

有了完美的战略定位蓝图&#xff0c;但如果不知道如何执行&#xff0c;那一切都是徒劳。今天&#xff0c;我们将揭示从策略到执行的战略定位秘密路径。首先我们先明确一下战略定位的相关概念以及实施战略定位的用途。 战略定位是什么意思? 战略定位可以视为企业在市场或竞争环…

Jetpack Compose 教程

一、简介 Jetpack Compose 是用于构建原生 Android 界面的新工具包。它使用更少的代码、强大的工具和直观的 Kotlin API&#xff0c;可以帮助您简化并加快 Android 界面开发。 在本教程中&#xff0c;您将使用声明性的函数构建一个简单的界面组件。您无需修改任何 XML 布局&am…

读高性能MySQL(第4版)笔记04_操作系统和硬件优化

1. 从软件本身和它运行的典型工作负载来看&#xff0c;MySQL通常也更适合运行在廉价硬件上 2. 基本资源 2.1. CPU 2.2. 内存 2.3. 磁盘 2.4. 瓶颈 2.5. 网络资源 3. CPU 3.1. 最常见的瓶颈是CPU耗尽 3.2. 检查CPU使用率来确定工作负载是否受CPU限制 3.3. 低延迟&…

JavaScipt中如何实现函数缓存?函数缓存有哪些场景?

1、函数缓存是什么&#xff1f; 函数缓存就是将函数运行的结果进行缓存。本质上就是用空间&#xff08;缓存存储&#xff09;换时间&#xff08;计算过程&#xff09; 常用于缓存数据计算结果和缓存对象。 缓存只是一个临时的数据存储&#xff0c;它保存数据&#xff0c;以便将…

异步编程 - 09 Spring框架中的异步执行_@Async注解异步执行原理源码解析

文章目录 概述小结好文推荐 概述 在Spring中调用线程将在调用含有Async注释的方法时立即返回&#xff0c;Spring是如何做到的呢&#xff1f;其实是其对标注Async注解的类做了代理&#xff0c;比如下面的类Async-AnnotationExample。 public class AsyncAnnotationExample {As…