如何在JavaScript中区分深层副本和浅层副本

by Lukas Gisder-Dubé

卢卡斯·吉斯杜比(LukasGisder-Dubé)

如何在JavaScript中区分深层副本和浅层副本 (How to differentiate between deep and shallow copies in JavaScript)

New is always better!

新总是更好!

You have most certainly dealt with copies in JavaScript before, even if you didn’t know it. Maybe you have also heard of the paradigm in functional programming that you shouldn’t modify any existing data. In order to do that, you have to know how to safely copy values in JavaScript. Today, we’ll look at how to do this while avoiding the pitfalls!

即使您不知道,您肯定也曾经用JavaScript处理过副本。 也许您还听说过函数式编程的范例,您不应修改任何现有数据。 为此,您必须知道如何安全地在JavaScript中复制值。 今天,我们将研究如何避免陷阱!

First of all, what is a copy?

首先,什么是副本?

A copy just looks like the old thing, but isn’t. When you change the copy, you expect the original thing to stay the same, whereas the copy changes.

副本看起来像旧的东西,但不是。 更改副本时,您希望原始内容保持不变,而副本会更改。

In programming, we store values in variables. Making a copy means that you initiate a new variable with the same value(s). However, there is a big potential pitfall to consider: deep copying vs. shallow copying. A deep copy means that all of the values of the new variable are copied and disconnected from the original variable. A shallow copy means that certain (sub-)values are still connected to the original variable.

在编程中,我们将值存储在变量中。 进行复制意味着您将启动一个具有相同值的新变量。 但是,有一个潜在的陷阱需要考虑: 深层复制浅层复制 。 深拷贝意味着新变量的所有值都将被复制并与原始变量断开连接 。 浅表副本意味着某些(子)值仍连接到原始变量。

To really understand copying, you have to get into how JavaScript stores values.
要真正理解复制,您必须了解JavaScript如何存储值。

原始数据类型 (Primitive data types)

Primitive data types include the following:

基本数据类型包括以下内容:

  • Number — e.g. 1

    数字-例如1

  • String — e.g. 'Hello'

    字符串—例如'Hello'

  • Boolean — e.g. true

    布尔值—例如, true

  • undefined

    undefined

  • null

    null

When you create these values, they are tightly coupled with the variable they are assigned to. They only exist once. That means you do not really have to worry about copying primitive data types in JavaScript. When you make a copy, it will be a real copy. Let’s see an example:

创建这些值时,它们会与分配给它们的变量紧密耦合。 它们仅存在一次。 这意味着您实际上不必担心在JavaScript中复制原始数据类型。 制作副本时,它将是真实副本。 让我们来看一个例子:

const a = 5
let b = a // this is the copy
b = 6
console.log(b) // 6
console.log(a) // 5

By executing b = a , you make the copy. Now, when you reassign a new value to b, the value of b changes, but not of a.

通过执行b = a ,您可以制作副本。 现在,当你重新分配一个新值b ,值b的变化,但不是a

复合数据类型-对象和数组 (Composite data types — Objects and Arrays)

Technically, arrays are also objects, so they behave in the same way. I will go through both of them in detail later.

从技术上讲,数组也是对象,因此它们的行为方式相同。 稍后,我将详细介绍它们。

Here it gets more interesting. These values are actually stored just once when instantiated, and assigning a variable just creates a pointer (reference) to that value.

在这里,它变得更加有趣。 这些值实际上在实例化时只存储一次,分配一个变量只会创建指向该值的指针(引用)

Now, if we make a copy b = a , and change some nested value in b, it actually changes a’s nested value as well, since a and b actually point to the same thing. Example:

现在,如果我们制作一个副本b = a ,并更改b某个嵌套值,则实际上也会更改a的嵌套值,因为ab实际上指向同一事物。 例:

const a = {
en: 'Hello',
de: 'Hallo',
es: 'Hola',
pt: 'Olà'
}
let b = a
b.pt = 'Oi'
console.log(b.pt) // Oi
console.log(a.pt) // Oi

In the example above, we actually made a shallow copy. This is often times problematic, since we expect the old variable to have the original values, not the changed ones. When we access it, we sometimes get an error. It might happen that you try to debug it for a while before you find the error, since a lot of developers do not really grasp the concept and do not expect that to be the error.

在上面的示例中,我们实际上制作了一个浅表副本 。 这通常是有问题的,因为我们期望旧变量具有原始值,而不是更改后的值。 当我们访问它时,有时会出现错误。 您可能会尝试在发现错误之前先对其进行调试,因为许多开发人员并未真正掌握该概念,也不希望这是错误。

Let’s have a look at how we can make copies of objects and arrays.

让我们看一下如何制作对象和数组的副本。

对象 (Objects)

There are multiple ways to make copies of objects, especially with the new expanding and improving JavaScript specification.

有多种方法可以复制对象,特别是在新的扩展和改进JavaScript规范中。

点差运算符 (Spread operator)

Introduced with ES2015, this operator is just great, because it is so short and simple. It ‘spreads’ out all of the values into a new object. You can use it as follows:

ES2015引入了该运算符,它非常简短,非常棒,它很棒。 它将所有值“散布”到一个新对象中。 您可以按以下方式使用它:

const a = {
en: 'Bye',
de: 'Tschüss'
}
let b = {...a}
b.de = 'Ciao'
console.log(b.de) // Ciao
console.log(a.de) // Tschüss

You can also use it to merge two objects together, for example const c = {...a, ...b} .

您还可以使用它来合并两个对象,例如const c = {...a, ...b}

对象分配 (Object.assign)

This was mostly used before the spread operator was around, and it basically does the same thing. You have to be careful though, as the first argument in the Object.assign() method actually gets modified and returned. So make sure that you pass the object to copy at least as the second argument. Normally, you would just pass an empty object as the first argument to prevent modifying any existing data.

这主要是在散布运算符出现之前使用的,基本上可以完成相同的操作。 但是,您必须要小心,因为Object.assign()方法中的第一个参数实际上已修改并返回。 因此,请确保至少传递对象以进行复制,并将其作为第二个参数。 通常,您只需将一个空对象作为第一个参数,以防止修改任何现有数据。

const a = {
en: 'Bye',
de: 'Tschüss'
}
let b = Object.assign({}, a)
b.de = 'Ciao'
console.log(b.de) // Ciao
console.log(a.de) // Tschüss

陷阱:嵌套对象 (Pitfall: Nested Objects)

As mentioned before, there is one big caveat when dealing with copying objects, which applies to both methods listed above. When you have a nested object (or array) and you copy it, nested objects inside that object will not be copied, since they are only pointers / references. Therefore, if you change the nested object, you will change it for both instances, meaning you would end up doing a shallow copy again. Example:// BAD EXAMPLE

如前所述,在处理复制对象时有一个很大的警告,它适用于上面列出的两种方法。 当您有一个嵌套对象(或数组)并复制它时,该对象内部的嵌套对象将不会被复制,因为它们只是指针/引用。 因此,如果您更改嵌套对象,则将在两个实例中都对其进行更改,这意味着您最终将再次进行浅表复制 。 例子//坏例子

const a = {
foods: {
dinner: 'Pasta'
}
}
let b = {...a}
b.foods.dinner = 'Soup' // changes for both objects
console.log(b.foods.dinner) // Soup
console.log(a.foods.dinner) // Soup

To make a deep copy of nested objects, you would have to consider that. One way to prevent that is manually copying all nested objects:

要制作嵌套对象深层副本 ,您必须考虑到这一点。 防止这种情况的一种方法是手动复制所有嵌套对象:

const a = {
foods: {
dinner: 'Pasta'
}
}
let b = {foods: {...a.foods}}
b.foods.dinner = 'Soup'
console.log(b.foods.dinner) // Soup
console.log(a.foods.dinner) // Pasta

In case you were wondering what to do when the object has more keys than only foods , you can use the full potential of the spread operator. When passing more properties after the ...spread , they overwrite the original values, for example const b = {...a, foods: {...a.foods}} .

如果您想知道当对象具有比foods更多的键时该怎么做,则可以利用散布算子的全部潜力。 在...spread之后传递更多属性时,它们将覆盖原始值,例如const b = {...a, foods: {...a.foods}}

深思熟虑地进行深拷贝 (Making deep copies without thinking)

What if you don’t know how deep the nested structures are? It can be very tedious to manually go through big objects and copy every nested object by hand. There is a way to copy everything without thinking. You simply stringify your object and parse it right after:

如果您不知道嵌套结构的深度怎么办? 手动浏览大对象并用手复制每个嵌套对象可能非常繁琐。 有一种无需思考即可复制所有内容的方法。 您只需将对象stringify并在之后parse它:

const a = {
foods: {
dinner: 'Pasta'
}
}
let b = JSON.parse(JSON.stringify(a))
b.foods.dinner = 'Soup'
console.log(b.foods.dinner) // Soup
console.log(a.foods.dinner) // Pasta

Here, you have to consider that you will not be able to copy custom class instances, so you can only use it when you copy objects with native JavaScript values inside.

在这里,您必须考虑到您将无法复制自定义类实例,因此仅当您复制内部具有JavaScript值的对象时才能使用它。

数组 (Arrays)

Copying arrays is just as common as copying objects. A lot of the logic behind it is similar, since arrays are also just objects under the hood.

复制数组与复制对象一样普遍。 它背后的很多逻辑都是相似的,因为数组也只是底层的对象。

点差运算符 (Spread operator)

As with objects, you can use the spread operator to copy an array:

与对象一样,您可以使用spread运算符复制数组:

const a = [1,2,3]
let b = [...a]
b[1] = 4
console.log(b[1]) // 4
console.log(a[1]) // 2

数组功能-映射,过滤,缩小 (Array functions — map, filter, reduce)

These methods will return a new array with all (or some) values of the original one. While doing that, you can also modify the values, which comes in very handy:

这些方法将返回一个新数组,其中包含原始数组的所有(或某些)值。 在此过程中,您还可以修改值,这非常方便:

const a = [1,2,3]
let b = a.map(el => el)
b[1] = 4
console.log(b[1]) // 4
console.log(a[1]) // 2

Alternatively you can change the desired element while copying:

或者,您可以在复制时更改所需的元素:

const a = [1,2,3]
const b = a.map((el, index) => index === 1 ? 4 : el)
console.log(b[1]) // 4
console.log(a[1]) // 2

数组切片 (Array.slice)

This method is normally used to return a subset of the elements, starting at a specific index and optionally ending at a specific index of the original array. When using array.slice() or array.slice(0) you will end up with a copy of the original array.

此方法通常用于返回元素的子集,该元素的子集从原始数组的特定索引开始,并且可选地终止于原始数组的特定索引。 当使用array.slice()array.slice(0)您将得到原始数组的副本。

const a = [1,2,3]
let b = a.slice(0)
b[1] = 4
console.log(b[1]) // 4
console.log(a[1]) // 2

嵌套数组 (Nested arrays)

Similar to objects, using the methods above to copy an array with another array or object inside will generate a shallow copy. To prevent that, also use JSON.parse(JSON.stringify(someArray)) .

与对象相似,使用上述方法将一个数组复制到另一个数组或对象内部将生成一个浅表副本 。 为了防止这种情况,还请使用JSON.parse(JSON.stringify(someArray))

奖励:复制自定义类的实例 (BONUS: copying instance of custom classes)

When you are already a pro in JavaScript and you deal with your custom constructor functions or classes, maybe you want to copy instances of those as well.

当您已经是JavaScript专业人士并且要处理自定义构造函数或类时,也许您也想复制这些实例。

As mentioned before, you cannot just stringify + parse those, as you will lose your class methods. Instead, you would want to add a custom copy method to create a new instance with all of the old values. Let’s see how that works:

如前所述,您不能仅仅对它们进行字符串化+解析,否则您将丢失类方法。 相反,您可能想添加一个自定义copy方法来创建一个具有所有旧值的新实例。 让我们看看它是如何工作的:

class Counter {
constructor() {
this.count = 5
}
copy() {
const copy = new Counter()
copy.count = this.count
return copy
}
}
const originalCounter = new Counter()
const copiedCounter = originalCounter.copy()
console.log(originalCounter.count) // 5
console.log(copiedCounter.count) // 5
copiedCounter.count = 7
console.log(originalCounter.count) // 5
console.log(copiedCounter.count) // 7

To deal with objects and arrays that are referenced inside of your instance, you would have to apply your newly learned skills about deep copying! I will just add a final solution for the custom constructor copy method to make it more dynamic:

要处理实例内部引用的对象和数组,您必须应用关于深度复制的新知识! 我将为自定义构造函数的copy方法添加最终解决方案,以使其更加动态:

With that copy method, you can put as many values as you want in your constructor, without having to manually copy everything!

使用该复制方法,您可以在构造函数中放置任意数量的值,而无需手动复制所有内容!

About the Author: Lukas Gisder-Dubé co-founded and led a startup as CTO for 1 1/2 years, building the tech team and architecture. After leaving the startup, he taught coding as Lead Instructor at Ironhack and is now building a Startup Agency & Consultancy in Berlin. Check out dube.io to learn more.

关于作者:LukasGisder-Dubé与他人共同创立并领导了一家初创公司担任CTO长达1 1/2年,建立了技术团队和架构。 离开创业公司后,他在Ironhack担任首席讲师的编码课程,现在正在柏林建立创业公司和咨询公司。 查看dube.io了解更多信息。

翻译自: https://www.freecodecamp.org/news/copying-stuff-in-javascript-how-to-differentiate-between-deep-and-shallow-copies-b6d8c1ef09cd/

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

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

相关文章

网站QQ全屏PHP代码,QQ技术导航升级版 超级导航美化版带后台版 PHP源码

QQ技术导航升级版 超级导航美化版带后台版改进F2样式,主针对QQ教程网、卡盟、博客、提供更好收录的位置。改进QQ技术导航背景,增加整体美观效果。去掉死链页面,站长操作使用更加有扩大空间。优化后台登陆界面,去掉织梦后台携带的广…

MySQL基础操作(一)

MySQL操作 一、创建数据库 # utf-8 CREATE DATABASE 数据库名称 DEFAULT CHARSET utf8 COLLATE utf8_general_ci;# gbk CREATE DATABASE 数据库名称 DEFAULT CHARACTER SET gbk COLLATE gbk_chinese_ci; 二、用户管理 创建用户create user 用户名IP地址 identified by 密码; 删…

集合框架05

一、HashSet集合 1 public class Demo01 {2 /*3 * Set接口,特点不重复元素,没索引4 * Set接口的实现类,HashSet(哈希表)5 * 特点:无序集合,存储和取出的顺序不同,没有索引,不…

leetcode1233. 删除子文件夹

你是一位系统管理员,手里有一份文件夹列表 folder,你的任务是要删除该列表中的所有 子文件夹,并以 任意顺序 返回剩下的文件夹。 我们这样定义「子文件夹」: 如果文件夹 folder[i] 位于另一个文件夹 folder[j] 下,那…

HIVE-分桶表的详解和创建实例

我们学习一下分桶表,其实分区和分桶这两个概念对于初学者来说是比较难理解的。但对于理解了的人来说,发现又是如此简单。 我们先建立一个分桶表,并尝试直接上传一个数据 create table student4(sno int,sname string,sex string,sage int, sd…

51nod1270(dp)

题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId1270 题意:中文题诶~ 思路:dp sabs(a1-a0)abs(a2-a1).... 要使s尽量大,需要让abs(ai-ai-1)尽量大,那么可以让其中一个尽量小&…

Windows IIS 日志分析研究(Log Parser Log Parser Lizard Log Parser Studio) update...

Windows主要有以下三类日志记录系统事件:应用程序日志、系统日志和安全日志。 存放目录:X:\Windows\System32\winevt\Logs\ System.evtx 系统日志 Application.evtx 应用程序日志 Security.evtx 安全日志 审核策略与事件查看器 # 管理工具 → 本地安全…

ios php ide,最好的PHP IDE for Mac? (最好免费!)

这里是PHP的Mac IDE的下降NetBeans自由!此外,所有产品的最佳功能。包括内联数据库连接,代码完成,语法检查,颜色编码,分割视图等。下降:这是一个内存猪在Mac上。准备好允许一半的内存&#xff0c…

leetcode79. 24 点游戏

你有 4 张写有 1 到 9 数字的牌。你需要判断是否能通过 *,/,,-,(,) 的运算得到 24。 示例 1: 输入: [4, 1, 8, 7] 输出: True 解释: (8-4) * (7-1) 24 代码 class Solution {public boolean judgePoint24(int[] n…

Linux邮件系统整合windows 2008 R2 AD域认证更新

1. 安装只要执行install.sh即可。(安装包约40几M) 2.文档更新功能 (原v1.0文档链接:http://godoha.blog.51cto.com/108180/691376) 本文转自 godoha 51CTO博客,原文链接:http://blog.51cto.com/…

004:神秘的数组初始化_使容器神秘化101:面向初学者的深入研究容器技术

004:神秘的数组初始化by Will Wang王Will 介绍 (Introduction) Regardless of whether you are a student in school, a developer at some company, or a software enthusiast, chances are you heard of containers. You may have also heard that containers are lightweig…

php js动态显示系统时间,PHP+JS动态显示当前时间

header("content-type:text/html;charsetgb2312");date_default_timezone_set("PRC");echo var dayNames new Array("星期日","星期一","星期二","星期三","星期四","星期五","星期六&…

代码整洁之道,clean code

一、注释 1、不准确的注释比没有注释更令人头疼 尽量用语义化的代码来解释你的意图,而不是依赖注释来解释一段代码 原因很简单:程序员不能坚持维护注释。 代码在后期维护中,不断的优化、变动,很有可能最初的注释已和现有的代码没…

java 获取手机归属地,引起net.UnknownHostException错误

这个问题是请求,重定向了,跟入源码。修改了地址,变成302 Connection connect Jsoup.connect(url);connect.header("Host", "http://info.bet007.com");connect.header("User-Agent", " Mozilla/5.0 (Wi…

leetcode713. 乘积小于K的子数组(双指针)

给定一个正整数数组 nums。 找出该数组内乘积小于 k 的连续的子数组的个数。 示例 1: 输入: nums [10,5,2,6], k 100 输出: 8 解释: 8个乘积小于100的子数组分别为: [10], [5], [2], [6], [10,5], [5,2], [2,6], [5,2,6]。 需要注意的是 [10,5,2] 并不是乘积小于100的子数…

Scrum Guides 2017年最新修改

采用Scrum中增加章节\\最初Scrum是为了管理与开发产品而开发的。从90年代早期开始,Scrum已经在全球范围内得到广泛应用:\\研究及识别可行的市场、技术与产品能力;\\t开发产品及增强功能;\\t每天多次频繁发布产品及增强功能&#x…

这是我最喜欢的使用React Native创建生产级应用程序的技巧

Trust me when I say this, React Native is hard. And it’s not the usual hard of what we think hard is. It is hard in terms of working with in general. In this blog post, I’ll go over some tips and tricks and eventually the best practices I’ve deployed fo…

HTTP 协议 -- 浏览器缓存机制

浏览器缓存机制浏览器缓存机制主要是 HTTP 协议定义的缓存机制。HTTP 协议中有关缓存的缓存信息头的关键字有 Cache-Control,Pragma,Expires,Last-Modified/ETag 等。浏览器请求流程浏览器第一请求流程:浏览器再次请求流程&#x…

php 获取实例的类名,PHP类名获取方式及单例模式实现

类名是什么意思?顾名思义就是各类起了一个名字,java中有两种数据类型,基本数据类型和引用数据类型,这里类就是引用数据类型,我们在定义一个类的时候必须给类起一个名字,一边后面的使用比如:int …

CAP理论的理解

CAP理论作为分布式系统的基础理论,它描述的是一个分布式系统在以下三个特性中: 一致性(Consistency)可用性(Availability)分区容错性(Partition tolerance)最多满足其中的两个特性。也就是下图所…