js实现多个对象深拷贝方法 ,js手写深拷贝函数

本文目录

JS实现深拷贝、instanceof、判断是否为数组


JS中拷贝对象可以按照拷贝的程度可以分为浅拷贝和深拷贝,有些时候我们需要拷贝之后的对象和拷贝之前的对象解耦,即脱离联系,也就是改变其中一者,另一者不会变化,典型的场景有:状态的回溯。如果我们对状态对象使用浅拷贝,则无法对状态进行回溯,但如果使用深拷贝,则可以很容易的对状态进行回溯和跟踪。实现深拷贝,主要由以下两种方式:(值得一提的是,JS原生数组中的 concat、slice 方法还有 Object.assign 方法都是一层拷贝,即浅拷贝)

instanceof 是用来判断对象是是某类的实例,或者是否是某类的子类的实例,它的实现原理可以是下面这样,L 表示实例,R 表示某类。

JS中数组也属于对象,所以无法通过 typeof 直接判断,这个时候就得采用其他方式,主要有一下几种方式:

方法 1-3 有兼容性问题,方法 5 可能不兼容老款浏览器,所以最好使用方法 4

js实现多个对象深拷贝方法
,js手写深拷贝函数图1

js实现深拷贝


基本数据类型: String、Number、Boolean、Null、Undefined
复杂数据类型: Object

使用typeof进行判断数据类型,只能够判断基本数据类型string number boolean 以及 function,而null和object不能够进一步的判断。

使用A instanceof B的方式进行判断,字面意思,A是否是B的实例,可以判断出Array和Object类型,但是undefined和null不能区分数据类型,基础的数据类型,因为不是使用new出来的,也测试不出来。

Object.prototype.toString()方法可以返回一个表示该对象的字符串'[object type]',为了每个对象都能通过 Object.prototype.toString() 来检测,需要以 Function.prototype.call() 或者 Function.prototype.apply() 的形式来调用,传递要检查的对象作为第一个参数,称为 thisArg。

实现思路:

其实还有很多深拷贝的方式,比如说Object.assign(),JSON.stringify()等方式,可以自己下去尝试一下哦!


js实现多个对象深拷贝方法
,js手写深拷贝函数图2

如何实现数组深拷贝和浅拷贝


如何实现数组深拷贝和浅拷贝?

1.背景介绍

Javascript分原始类型与引用类型。Array是引用类型,直接用“=”号赋值的话,只是把源数组的地址(或叫指针)赋值给目的数组,并没有实现数组的数据的拷贝。这种方式的实现属于浅拷贝。

深拷贝是开辟新的储存空间,两个对象对应两个不同的地址,修改一个对象的属性,不会改变另一个对象的属性。

2.知识剖析

一维数组的深拷贝方法:slice()和concat()

slice()的使用方法

slice()语法:arrayObj.slice(start,[end])

slice方法是通过参数start和end的传入值来返回数组中的一段,该方法不对原数组进行操作,而是返回一个子数组

start:必需。规定从何处开始选取。如果是负数,那么它规定从数组尾部开始算起的位置。也就是说,-1 指最后一个元素,-2 指倒数第二个元素,以此类推。

end:可选。规定从何处结束选取。该参数是数组片断结束处的数组下标。如果没有指定该参数,那么切分的数组包含从 start 到数组结束的所有元素。如果这个参数是负数,那么它规定的是从数组尾部开始算起的元素。

返回值:返回一个新的数组,包含从 start 到 end (不包括该元素)的 arrayObject 中的元素(如果 end 未被规定,那么 slice() 方法会选取从 start 到数组结尾的所有元素)。

concat()的使用方法

concat()语法:arrayObject.concat(arrayX,arrayX,......,arrayX)

arrayX:必需,可以是具体的值,也可以是数组对象。可以是任意多个。

concat() 方法用于连接两个或多个数组。 该方法不会改变现有的数组,而仅仅会返回一个新的数组。如果要进行 concat() 操作的参 数是数组,那么添加的是数组中的元素,而不是数组。

3.常见问题

1、jquery中数组深拷贝办法

语法:jQuery.extend( [deep ], target, object1 [, objectN ] )

将两个或更多对象的内容合并到第一个对象。

deep:可选。 Boolean类型 指示是否深度合并对象,默认为false。如果该值为true,且多个对象的某个同名属性也都是对象,则该"属性对象"的属性也将进行合并。

2、什么是深拷贝?

深拷贝:指的是拷贝一个对象时,不仅仅把对象的引用进行复制,还把该对象引用的值也一起拷贝。这样进行深拷贝后的拷贝对象就和源对象互相独立,其中任何一个对象的改动都不会对另外一个对象造成影响。举个例子,一个人叫张三,然后使用克隆技术以张三来克隆另外一个人叫李四,这样张三和李四就是相互独立的,不管张三缺胳膊还是李四少腿了都不会影响另外一个人。在.NET领域,值对象就是典型的例子,如int, Double以及结构体和枚举等。

3、什么是浅拷贝呢?

浅拷贝:指的是拷贝一个对象时,仅仅拷贝对象的引用进行拷贝,但是拷贝对象和源对象还是引用同一份实体。此时,其中一个对象的改变都会影响到另一个对象。例如,一个人一开始叫张三,后来改名字为张老三了,可是他们还是同一个人,不管张三缺胳膊还是张老三少腿,都反应在同一个人身上。在.NET中引用类型就是一个例子。

4 解决方案

jquery.extend()

语法:jQuery.extend( [deep ], target, object1 [, objectN ] )

将两个或更多对象的内容合并到第一个对象。

deep:可选。 Boolean类型 指示是否深度合并对象,默认为false。如果该值为true,且多个对象的某个同名属性也都是对象,则该"属性对象"的属性也将进行合并。

5.编码实战

6.扩展思考

slice和concat对数组深拷贝的局限性

slice和concat这两个方法,仅适用于对不包含引用对象的一维数组的深拷贝。对于数组内部存在对象和数组,当改变对象属性和内部数组的元素后,深拷贝的数组同样也发生了改变。

js实现多个对象深拷贝方法
,js手写深拷贝函数图3

JS中实现深拷贝的几种方法objectArray


4.通过第三方工具实现深拷贝

lodash.cloneDeep
数组深拷贝

1. concat(arr1, arr2,....)

2. slice(idx1, idx2)

参数可以省略

1)没有参数是拷贝数组

2)只有一个参数是从该位置起到结束拷贝数组元素

3)两个参数,拷贝从起始位置到结束位置的元素(不包含结束位置的元素:含头不含尾)

注意:当数组中的元素均为一维是深拷贝

          数组中元素一维以上是值的引用

js实现多个对象深拷贝方法
,js手写深拷贝函数图4

以上就是关于js实现多个对象深拷贝方法 ,js手写深拷贝函数的全部内容,以及js实现多个对象深拷贝方法 的相关内容,希望能够帮到您。

上一篇 2022年11月25日12时44分31秒
下一篇 2022年11月25日12时47分02秒

相关推荐

  • 为什么有时候使用不定冠词(a或an)的形式,有时候不使用

    在英语中,不定冠词(a或an)的使用是很重要的,它们用来指代一个未知或不确定的事物或对象。然而,在某些情况下,我们会发现有时候需要使用不定冠词,而有时候则不需要。那么,这是为什么呢? 首先,我们需要了解的是,不定冠词的使用是有规则的。当紧随其后的单词是以辅音字母开头的时候,我们就需要使用"a",例如:a dog,a cat。而当紧随其后的单词是以元音字母开头的时候,我们就需要使用"an",例如:an apple,an elephant。这是因为英语中的元音字母发音时需要发出气流,而辅音字母则不需要。 但是,在某些情况下,我们不需要使用不定冠词。例如,在使用专有名词时,我们通常不需要使用不定冠词…

    英语知识 2023年05月26日
    11
  • 高考英语作文亮点句型,高考英语很重要吗

    本文目录 1.高考英语很重要吗 2.高考英语作文亮点句型 必背高分句型 3.英语作文高级句型精选 让你离高分更近一步 高考英语很重要吗 首先,我想说的是,高考英语作文没有所谓的通用句型!不同的话题有不同的句型!提高高考英语作文并非没有办法。 在日常学习中,我们应该加强阅读和写作的训练。只有读,才能熟悉;只有熟悉,才能完美!2006年,我在初中教了一个学生。她每天用英语写日记,一次积累十几篇文章供我批评。每次我都会仔细发现她的单词、语法和文法的缺点,仔细写批语,鼓励她坚持下去。她在高中时仍然每个星期天都来找我批评。高中毕业成绩不理想,只考了三本书。复习那一年,她被我介绍到我们学校换班(2011年…

    2023年02月10日
    16
  • 简单将来时的缩写形式是什么

    英语中的简单将来时表示一个将来的动作或事件。通常情况下,我们使用“will + 动词原形”来表达简单将来时。但是在口语中,为了简化表达,我们经常使用简写形式来表示简单将来时。 简单将来时的缩写形式是“'ll”,这是由will缩写而来的。例如,我们可以说:“I'll go to the store tomorrow.”(我明天会去商店)。在这个例子中,“'ll”代替了完整的形式“will”。 除了“'ll”以外,还有一些其他的缩写形式,如“won't”(will not的缩写)和“'d”(would或had的缩写)。例如,“I'd love to go to the party tonight.…

    英语知识 2023年05月26日
    11
  • 简述一下英语的将来完成时态。

    英语将来完成时态是表达将来某个时间点之前已完成的动作的一种时态。这个时态通常用于描述某个将来的时间点之前已经完成的动作。 将来完成时态的构成方式是:将来时态的助动词“will”与完成时态的助动词“have”组合,再加上动词的过去分词形式。 举个例子,如果我们想要表达在明天晚上8点之前完成的动作,我们可以这样说:“By 8 o'clock tomorrow night, I will have finished my homework.”(在明天晚上8点之前,我会完成我的家庭作业。) 在英语中,将来完成时态通常用于以下几种情况: 预测未来某个时间点之前的完成动作。例如:“By the end o…

    英语知识 2023年05月26日
    11
  • 什么是方式状语从句

    在英语语法中,状语从句是指用来修饰主句谓语动词的从句。其中,方式状语从句是一种状语从句,用来描述主句中的动作或状态是如何发生的。 方式状语从句通常由“how”引导,表示“以何种方式”,或者由“as if/as though”引导,表示“仿佛”。例如: She explained how to use the new software. (她解释了如何使用新软件。) He ran as if he was being chased by a dog. (他跑的像是被狗追赶。) 方式状语从句可以出现在主句之前、之后或中间。它可以用来修饰主句中的动词、形容词、副词或整个句子。例如: She spok…

    英语知识 2023年05月26日
    11
  • 要是怎么样英语,要是就好了.用英语表达

    本文目录 1.要是就好了.用英语表达 2.他长得怎样的英文怎么写 要是就好了.用英语表达 答案:if only 例:If only I had a memory of yours.如果我有你的记忆力该多好呀! If only I had another chance.要是我再有一次机会就好了. 此词组是虚拟语气,表达不可能实现的事情,或者对错过的事情表示遗憾. 他长得怎样的英文怎么写 问题一:……怎么样?用英语怎么说 ……怎么样 What about.... How about... 如果满意,请记得采纳,谢谢(*^__^*) 嘻嘻 问题二:不管怎样英文怎么写 不管怎样 adv. anyhow…

    英语知识 2023年02月10日
    11
  • 转折连词的用法是什么

    转折连词是语法中一个重要的部分,它用于连接两个相对矛盾的句子或意思。在英语中,我们常用的转折连词有“but”、“however”、“although”、“nevertheless”等等。 首先,我们来看“but”这个转折连词。它常用于连接两个相对矛盾的句子,表示前后句之间的转折关系。例如,“I am tired, but I need to finish my work.”(我很累,但我需要完成我的工作。)在这个例句中,“but”连接了前后两个句子,表示在疲劳的情况下,仍然需要完成工作,两个句子之间存在矛盾。 其次,我们来看“however”这个转折连词。和“but”类似,它也表示前后句之间的…

    英语知识 2023年05月26日
    11
  • 现在完成时可以与多种时间状语一起使用,但需要注意哪些问题

    本文目录 1. 时间状语的选择 2. 时间状语的位置 3. 语态的选择 4. 句型的转换 现在完成时是英语中用于描述过去发生的动作或状态对现在造成影响的一种时态,通常使用have/has + 过去分词构成。而现在完成时也可以与多种时间状语一起使用,但需要注意以下几点。 1. 时间状语的选择 时间状语可以是一个具体的时间点,也可以是一个时间段。常见的时间状语有:already, yet, ever, never, just, recently, lately, recently, before, since, for,等等。使用时需要根据句子的需要选择合适的时间状语,以便更好地表达动作发生的时间…

    英语知识 2023年05月26日
    11