iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >JavaScript中深拷贝与浅拷贝实例运用
  • 923
分享到

JavaScript中深拷贝与浅拷贝实例运用

2023-06-29 03:06:15 923人浏览 安东尼
摘要

这篇文章主要介绍了javascript中深拷贝与浅拷贝实例运用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript中深拷贝与浅拷贝实例运用文章都会有所收获,下面我们一起来看看吧。1 浅拷贝概念深拷

这篇文章主要介绍了javascript中深拷贝与浅拷贝实例运用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript中深拷贝与浅拷贝实例运用文章都会有所收获,下面我们一起来看看吧。

1 浅拷贝概念

深拷贝和浅拷贝是只针对Object和Array这样的引用数据类型的。

浅拷贝是创建一个新对象,该对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型,拷贝的就是基本类型的值,如果属性是引用类型,拷贝的就是内存地址 ,所以如果其中一个对象改变了这个地址,就会影响到另一个对象

示例代码:

let people = { //定义一个People对象    name: "张三",    age: 3,    address: "中国"}console.log("原对象:", people);let newPeople = people; //进行浅拷贝console.log("新对象:", newPeople);//原对象: { name: '张三', age: 3, address: '中国' }//新对象: { name: '张三', age: 3, address: '中国' }//为对象修改名字newPeople.name = "橘猫吃不胖";console.log("原对象:", people);console.log("新对象:", newPeople);//原对象: { name: '橘猫吃不胖', age: 3, address: '中国' }//新对象: { name: '橘猫吃不胖', age: 3, address: '中国' }

从上面的示例可以看出,当newPeople的name属性修改后,原来的people也发生了变化,这是因为新创建的对象与旧对象具有相同的内存地址

2 深拷贝概念

深拷贝是将一个对象从内存中完整的拷贝一份出来,从堆内存中开辟一个新的区域存放新对象,且修改新对象不会影响原对象

示例代码:

let people = { //定义一个People对象    name: "张三",    age: 3,    address: "中国"}//对people进行深拷贝let newPeople = JSON.parse(jsON.stringify(people));console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '张三', age: 3, address: '中国' }//修改新对象中的adress属性newPeople.address = "俄罗斯";console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '张三', age: 3, address: '俄罗斯' }

从上面的例子可以看出,深拷贝后,修改新对象,不会影响原对象。

3 浅拷贝的实现方式

3.1 Object.assign()

Object.assign()方法用于将所有可枚举属性的值从一个或多个源对象分配到目标对象。它将返回目标对象。Object.assign() 进行的是浅拷贝,拷贝的是对象的属性的引用,而不是对象本身。

语法:

Object.assign(target, ...sources)//target:目标对象;sources:源对象。

如果目标对象中的属性具有相同的键,则属性将被源对象中的属性覆盖。后面的源对象的属性将类似地覆盖前面的源对象的属性。

示例:

const target = { a: 1, b: 2 };const source = { b: 4, c: 5 };//进行浅拷贝const returnedTarget = Object.assign(target, source);console.log(target);console.log(returnedTarget);// { a: 1, b: 4, c: 5 }// { a: 1, b: 4, c: 5 }//修改其中的值target.b = 10;console.log(target);console.log(returnedTarget);// { a: 1, b: 10, c: 5 }// { a: 1, b: 10, c: 5 }

当对象object只有一层的时候,是深拷贝,示例代码如下:

const obj = { name: "橘猫吃不胖" };//进行浅拷贝let newObj = Object.assign({}, obj);//修改新对象中的name属性为张三newObj.name = "张三";console.log("原对象:", obj);console.log("新对象:", newObj);// 原对象: { name: '橘猫吃不胖' }// 新对象: { name: '张三' }

3.2 Array.prototype.concat()

concat()方法用于合并两个或多个数组。此方法不会更改现有数组,而是返回一个新数组。

语法:

var new_array = old_array.concat(value1[, value2[, ...[, valueN]]])//valueN可选,数组和/或值,将被合并到一个新的数组中。//如果省略了所有 valueN 参数,则 concat 会返回调用此方法的现存数组的一个浅拷贝。

示例代码:

let arr1 = [1, 2, { name: "橘猫吃不胖" }];//进行浅拷贝let arr2 = arr1.concat();console.log("原数组:", arr1);console.log("新数组:", arr2);// 原数组: [ 1, 2, { name: '橘猫吃不胖' } ]// 新数组: [ 1, 2, { name: '橘猫吃不胖' } ]//修改原数组arr1[1] = "hhhhh";console.log("原数组:", arr1);console.log("新数组:", arr2);// 原数组: [ 1, 'hhhhh', { name: '橘猫吃不胖' } ]// 新数组: [ 1, 2, { name: '橘猫吃不胖' } ]

3.3 Array.prototype.slice()

slice() 方法返回一个新的数组对象,这一对象是一个由 begin 和 end 决定的原数组的浅拷贝(包括 begin,不包括end)。原始数组不会被改变。

语法:

arr.slice([begin[, end]])//begin:可选,提取起始处的索引(从 0 开始),从该索引开始提取原数组元素。//如果该参数为负数,则表示从原数组中的倒数第几个元素开始提取,slice(-2) 表示提取原数组中的倒数第二个元素到最后一个元素(包含最后一个元素)。//如果省略 begin,则 slice 从索引 0 开始。//如果 begin 超出原数组的索引范围,则会返回空数组。//end:可选,提取终止处的索引(从 0 开始),在该索引处结束提取原数组元素。slice 会提取原数组中索引从 begin 到 end 的所有元素(包含 begin,但不包含 end)。//slice(1,4) 会提取原数组中从第二个元素开始一直到第四个元素的所有元素 (索引为 1, 2, 3的元素)。//如果该参数为负数, 则它表示在原数组中的倒数第几个元素结束抽取。 slice(-2,-1) 表示抽取了原数组中的倒数第二个元素到最后一个元素(不包含最后一个元素,也就是只有倒数第二个元素)。//如果 end 被省略,则 slice 会一直提取到原数组末尾。//如果 end 大于数组的长度,slice 也会一直提取到原数组末尾。

示例代码:

let arr1 = [1, 2, { name: "橘猫吃不胖" }];//进行浅拷贝let arr2 = arr1.slice();console.log("原数组:", arr1);console.log("新数组:", arr2);// 原数组: [ 1, 2, { name: '橘猫吃不胖' } ]// 新数组: [ 1, 2, { name: '橘猫吃不胖' } ]//修改原数组arr1[1] = "hhhhh";console.log("原数组:", arr1);console.log("新数组:", arr2);// 原数组: [ 1, 'hhhhh', { name: '橘猫吃不胖' } ]// 新数组: [ 1, 2, { name: '橘猫吃不胖' } ]

3.4 直接赋值

直接使用“=”赋值可以实现浅拷贝,示例代码如下:

let obj1 = { //定义一个对象obj1    name: "张三",    age: 34}let obj2 = obj1; //进行浅拷贝console.log("obj1:", obj1);console.log("obj2:", obj2);// obj1: { name: '张三', age: 34 }// obj2: { name: '张三', age: 34 }//修改obj2中的name属性obj2.name = "橘猫吃不胖";console.log("obj1:", obj1);console.log("obj2:", obj2);// obj1: { name: '橘猫吃不胖', age: 34 }// obj2: { name: '橘猫吃不胖', age: 34 }

4 深拷贝的实现方式

4.1 JSON.parse(JSON.stringify())

JSON是一种语法,用来序列化对象、数组、数值、字符串、布尔值和 null 。它基于JavaScript语法,但与之不同:JavaScript不是JSON,JSON也不是JavaScript。

JSON对象包含两个方法:用于解析JSON的parse()方法,以及将对象/值转换为JSON字符串的stringify()方法,下面对这两种方法进行一些介绍。

JSON.parse()方法用来解析JSON字符串,构造由字符串描述的JavaScript值或对象。提供可选的 reviver 函数用以在返回之前对所得到的对象执行变换(操作)。

语法:

JSON.parse(text[, reviver])//text:要被解析成 JavaScript 值的字符串//reviver,可选,转换器, 如果传入该参数(函数),可以用来修改解析生成的原始值,调用时机在 parse 函数返回之前。

示例:

JSON.parse('{}');              // {}JSON.parse('true');            // trueJSON.parse('"foo"');           // "foo"JSON.parse('[1, 5, "false"]'); // [1, 5, "false"]JSON.parse('null');            // null

JSON.stringify()方法将一个 JavaScript 对象或值转换为 JSON 字符串,如果指定了一个 replacer 函数,则可以选择性地替换值,或者指定的 replacer 是数组,则可选择性地仅包含数组指定的属性。

语法:

JSON.stringify(value[, replacer [, space]])//value:将要序列化成 一个 JSON 字符串的值。//replacer,可选,如果该参数是一个函数,则在序列化过程中,被序列化的值的每个属性都会经过该函数的转换和处理;如果该参数是一个数组,则只有包含在这个数组中的属性名才会被序列化到最终的 JSON 字符串中;如果该参数为 null 或者未提供,则对象所有的属性都会被序列化。//space,可选,指定缩进用的空白字符串,用于美化输出(pretty-print);如果参数是个数字,它代表有多少的空格;上限为10。该值若小于1,则意味着没有空格;如果该参数为字符串(当字符串长度超过10个字母,取其前10个字母),该字符串将被作为空格;如果该参数没有提供(或者为 null),将没有空格。

示例:

JSON.stringify({});                        // '{}'JSON.stringify(true);                      // 'true'JSON.stringify("foo");                     // '"foo"'JSON.stringify([1, "false", false]);       // '[1,"false",false]'JSON.stringify({ x: 5 });                  // '{"x":5}'

深拷贝示例代码:

let people = { //定义一个People对象    name: "张三",    age: 3,    address: "中国"}//对people进行深拷贝let newPeople = JSON.parse(JSON.stringify(people));console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '张三', age: 3, address: '中国' }//修改新对象中的adress属性newPeople.address = "俄罗斯";console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '张三', age: 3, address: '俄罗斯' }

用JSON.stringify将对象转成JSON字符串,再用JSON.parse()把字符串解析成对象,这样新的对象产生了,实现深拷贝。这种方法虽然可以实现数组或对象深拷贝,但不能处理函数。

let arr1 = [1, 2, { name: "橘猫吃不胖" }, function () { }];//进行深拷贝let arr2 = JSON.parse(JSON.stringify(arr1));console.log("原数组:", arr1);console.log("新数组:", arr2);// 原数组: [ 1, 2, { name: '橘猫吃不胖' }, [Function (anonymous)] ]// 新数组: [ 1, 2, { name: '橘猫吃不胖' }, null ]

由上面例子可以看出,函数并没有被拷贝在arr2中。这是因为 JSON.stringify() 方法是将一个JavaScript值(对象或者数组)转换为一个 JSON字符串,不能接受函数。

4.2 函数库lodash

Lodash是一个JavaScript库,提供了多个实用程序功能,而Lodash库中最常用的功能之一是cloneDeep()方法。此方法有助于深度克隆对象,还可以克隆JSON.stringify()方法的局限性,即不可序列化的属性。

示例代码:

const lodash = require("lodash");let people = {    name: "张三",    age: 3,    address: "中国"}//对people进行深拷贝let newPeople = lodash.cloneDeep(people);console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '张三', age: 3, address: '中国' }//修改新对象中的adress属性newPeople.address = "俄罗斯";newPeople.name = "橘猫吃不胖";console.log("原对象:", people);console.log("新对象:", newPeople);// 原对象: { name: '张三', age: 3, address: '中国' }// 新对象: { name: '橘猫吃不胖', age: 3, address: '俄罗斯' }

关于“JavaScript中深拷贝与浅拷贝实例运用”这篇文章的内容就介绍到这里,感谢各位的阅读!相信大家对“JavaScript中深拷贝与浅拷贝实例运用”知识都有一定的了解,大家如果还想学习更多知识,欢迎关注编程网精选频道。

--结束END--

本文标题: JavaScript中深拷贝与浅拷贝实例运用

本文链接: https://www.lsjlt.com/news/322119.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • JavaScript中深拷贝与浅拷贝实例运用
    这篇文章主要介绍了JavaScript中深拷贝与浅拷贝实例运用的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇JavaScript中深拷贝与浅拷贝实例运用文章都会有所收获,下面我们一起来看看吧。1 浅拷贝概念深拷...
    99+
    2023-06-29
  • 浅拷贝与深拷贝
       名词解释 1.对象:被分配的一块内存,存储其所代表的值 2.引用:是自动形成的从变量到对象的指针 3.注意:类型(int类型,long类型(python3已去除long类型,只剩下int类型的数据))属于对象,不是变量 4.不可变...
    99+
    2023-01-30
  • JavaScript中深拷贝与浅拷贝详解
    目录1 浅拷贝概念2 深拷贝概念3 浅拷贝的实现方式3.1 Object.assign()3.2 Array.prototype.concat()3.3 Array.pro...
    99+
    2024-04-02
  • JavaScript浅拷贝与深拷贝如何实现
    这篇文章主要介绍“JavaScript浅拷贝与深拷贝如何实现”的相关知识,小编通过实际案例向大家展示操作过程,操作方法简单快捷,实用性强,希望这篇“JavaScript浅拷贝与深拷贝如何实现”文章能帮助大家解决问题。在 JavaScript...
    99+
    2023-06-27
  • JavaScript深拷贝与浅拷贝实现详解
    目录对于基本类型数据对于引用类型数据实现深拷贝简单版够用版structuredClone对于基本类型数据 可以说都是深拷贝。 对于引用类型数据 对于引用类型数据,浅拷贝 后,因为浅拷...
    99+
    2022-11-13
    JavaScript深拷贝与浅拷贝 JS深拷贝与浅拷贝
  • JavaScript深拷贝与浅拷贝是什么
    这篇文章主要介绍了JavaScript深拷贝与浅拷贝是什么,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。1 浅拷贝概念深拷贝和浅拷贝是只针对Object和Array这样的引用...
    99+
    2023-06-29
  • python深拷贝与浅拷贝
    可变对象与不可变对象 要理解深拷贝和浅拷贝,首先要理解可变对象和不可变对象。 不可变对象:该对象所指向的内存中的值不能被改变,修改对象的值时,由于其指向的值不能被改变,因此实际上是在内存中重新开辟一个地址用来存储新的值,然后将对象指向这个...
    99+
    2023-01-30
    python
  • 浅谈JavaScript浅拷贝和深拷贝
    目录一、直接赋值二、浅拷贝三、深拷贝1. JSON对象的方式2. 递归复制网上关于这个话题,讨论有很多了,根据各路情况我自己整理了一下,最后还是能接近完美的实现深拷贝,欢迎大家讨论。...
    99+
    2024-04-02
  • Python深拷贝与浅拷贝引用
    目录(1)、存在父对象和子对象(2)、如果只存在父对象前言: 在Python中,对象赋值在本质上是对对象的引用,当创建一个对象把它赋值给另一个变量的时候,Python并没有拷贝这个对...
    99+
    2024-04-02
  • 浅拷贝&深拷贝
    浅拷贝新的对象指向原来对象的地址 深拷贝新的对象中,原来是可变对象,会新复制一份值指向新的地址[11,22,33]若原来的对象里含有可变对象,里面的这个可变对象也会指向新的地址['qwer', 123, [44,55]] 参考:https...
    99+
    2023-01-30
  • python之浅拷贝与深拷贝
    浅拷贝是对于一个对象的顶层拷贝通俗的理解是:拷贝了引用,并没有拷贝内容 In [10]: a = [11,22,33] In [11]: b = a In [12]: id(a) Out[12]: 140343572333832 I...
    99+
    2023-01-31
    python
  • 老生常谈JavaScript深拷贝与浅拷贝
    目录1 浅拷贝概念2 深拷贝概念3 浅拷贝的实现方式3.1 Object.assign()3.2 Array.prototype.concat()3.3 Array.pro...
    99+
    2024-04-02
  • python深拷贝浅拷贝
    python深拷贝和浅拷贝问题:   什么是深拷贝?     (个人理解)深拷贝就是将原有的数据一模一样的拷贝一份,然后存到另一个地址中,而不是引用地址   什么是浅拷贝?     (个人理解)就是引用地址 (1)用等于号的拷贝都属于浅拷...
    99+
    2023-01-30
    python
  • 分析JavaScript浅拷贝和深拷贝
    本篇内容主要讲解“分析JavaScript浅拷贝和深拷贝”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“分析JavaScript浅拷贝和深拷贝”吧!一、直接赋值对象是引用类型,如果直接赋值给另外一...
    99+
    2023-06-25
  • 怎么理解Javascript深拷贝与浅拷贝
    怎么理解Javascript深拷贝与浅拷贝,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。 前言在 javascript &nbs...
    99+
    2024-04-02
  • Golang中深拷贝与浅拷贝详解
    目录什么是深拷贝?什么是浅拷贝?示例代码小结什么是深拷贝? 深拷贝(Deep Copy)是指原对象与拷贝的新对象互相独立,对其中任何一个对象的改动都不会对另外一个对象造成影响。值类型...
    99+
    2023-05-19
    Golang 深拷贝与浅拷贝 Golang深拷贝 Golang浅拷贝 Golang拷贝
  • JavaScript深拷贝与浅拷贝原理深入探究
    目录一、JS中数据的存储形式-堆栈二、深浅拷贝的三种方式遍历赋值Object.create()遍历赋值实现深拷贝一、JS中数据的存储形式-堆栈 我们先简单理解一下堆栈分别是啥: 什么...
    99+
    2022-11-13
    JS 深拷贝 浅拷贝 JS 深拷贝 JS 浅拷贝
  • 如何理解JavaScript中的浅拷贝与深拷贝
    本篇文章给大家分享的是有关如何理解JavaScript中的浅拷贝与深拷贝,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。 浅拷贝在使用JavaScript对数组进行操作...
    99+
    2023-06-16
  • javascript中怎么区分浅拷贝和深拷贝并实现深拷贝
    这篇文章将为大家详细讲解有关javascript中怎么区分浅拷贝和深拷贝并实现深拷贝,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。什么是拷贝 一个东西的拷贝看起来像是原来...
    99+
    2024-04-02
  • python-浅拷贝(copy)与深拷贝(deepcopy)
    一:对象与元素、可变与不可变:        (1)对象:对象有存储地址id【就好像某个地址的快递仓库】,对象中的元素都有一个地址id【就像仓库的某某货架】        (2)对象与对象的元素:(对象)或(对象中的元素)有不同的类型【数字...
    99+
    2023-09-03
    数学建模
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作