浅谈深浅拷贝|手摸手带你入坑

2023-11-07 02:10
文章标签 浅谈 拷贝 深浅 摸手

本文主要是介绍浅谈深浅拷贝|手摸手带你入坑,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

再次谈及深拷贝,已经过了两三年了!花有重开日人无再少年,从当初的懵懵懂懂到现在的油腻大叔,害

基本类型 与 引用类型

在这里我们先说明 基本类型引用类型 的区别

基本数据类型:直接存储在栈 (stack) 中的数据

String, Number, Boolean, Null, Undefined,Symbol

let a = 1
let b = a
b = 2
console.log(a,b)
// 1 , 2

ab 变量 都是基本类型,我们直接修改 b ,a 是不会被影响到的

引用数据类型:存储的是该对象在栈中引用,真实的数据存储在堆(heap)

Object 、Array 、Function 、Data

题来!!

let obj = {name: '严家辉',age: 18
}
let obj1 = {name: '严家辉',age: 18
}
console.log(obj === obj1)

输出什么?

相信有一部分同学已经知道了是 false,那是为什么呢,真相就是引用地址不同因为是引用类型

引用类型的变量,== 和 === 只会判断引用的地址是否相同,而不会判断对象具体里属性以及值是否相同

如何比较一个对象是否相等?

1.JSON.stringify

console.log(JSON.stringify(obj) === JSON.stringify(obj1))
// true

我们现在是将obj 转成了 string 类型,不会对比引用地址

'{"name":"严家辉","age":18}' === ' {"age":18,"name":"严家辉"}'

但是这样有一个问题

let obj = {name: '严家辉',age: 18
}
let obj1 = {age: 18,name: '严家辉'
}
console.log(JSON.stringify(obj) === JSON.stringify(obj1))

obj1 的顺序修改了一遍之后?现在这两个对象是否相等?还是这样转为字符串,肯定返回false

那么我们需要如何判断两个对象(不定顺序的kv)是否相等呢?

let obj = {name: '严家辉',age: 18
}
let obj1 = {age: 18,name: '严家辉'
}
const isSame = (obj1, obj2) => {var obj1keys = Object.keys(obj1);var obj2keys = Object.keys(obj2);if (obj2keys.length !== obj1keys.length)return falsefor (let i = 0; i <= obj1keys.length - 1; i++) {let key = obj1keys[i]if (!obj2keys.includes(key)) return falseif (obj2[key] !== obj1[key]) return false}return true
}
console.log(isSame(obj,obj1)) // true

这样对象值为基本数据类型的不管是顺序是否一致都可以对比了

关于引用地址

好了,我们回过头来看看这个

let obj = {name: '严家辉',age: 18
}
let obj1 = obj
console.log(obj === obj1)

我们在之前说过引用类型的 == 、=== 只是判断它的引用地址是否相同

那么它在这里的引用地址肯定是一样的,故打印 true

那么我现在需要修改 obj1age为 24

let obj = {name: '严家辉',age: 18
}
let obj1 = obj
obj1.age = 24
console.log('obj',obj)
console.log('obj1',obj1)

输出什么呢?

为啥obj也会改变呢

当我们使用=将这些变量赋值到另外的变量,实际上是将对应的值拷贝了一份,然后赋值给新的变量。

对象是通过引用传递,而不是值传递。也就是说,变量赋值只会将地址传递过去。

故我们修改 obj1 其实也是修改的 obj 本身

那么问题来了,假如我们在项目上面有个这样的需求

也就是我们目标对象 data ,a 函数先执行,然后执行b函数,但是b函数的要用到 data.name 为严家辉

let data = {name: '严家辉',age: 18
}
const a = () => data.name = '老严'
const b = () => console.log('b函数',data.name) // 老严
a()
b()

在这里我们就需要了解一下深拷贝了

什么是深拷贝?

在图中我们可以看到我们在内存中新开了一个堆用来拷贝obj的数据,但是修改obj1不会影响obj

建一个新的对象或数组,将原对象的各项属性的“值”(数组的所有元素)拷贝过来,是“数据”而不是“引用地址” 我们希望在改变新的对象的时候,不影响原对象

怎么实现呢?

1、JSON 序列化

let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {// 核心内容let data1 = JSON.parse(JSON.stringify(data))data1.name = '老严'data1.other.gender = '女'console.log('a函数',data1.name)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

先通过 JSON.stringify 将对象转为字符串再重新序列化为对象。

这个应该是最简单的深拷贝了

缺点是如果对象中包含函数会丢失

let data = {name: '严家辉',age: 18,other: {gender: "男"},// + 个test函数test: function() {}
}
const a = () => {let data1 = JSON.parse(JSON.stringify(data))data1.name = '老严'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

看看 test 凉了没

2、for in 遍历

const deepCopy = obj => {// 判断是数组还是对象let result = typeof obj.splice === "function" ? [] : {};if (obj && typeof obj === 'object') {for (let key in obj) {if (obj[key] && typeof obj[key] === 'object') {//如果对象的属性值为object的时候,递归调用deepClone,即在吧某个值对象复制一份到新的对象的对应值中。result[key] = deepCopy(obj[key]);} else {//如果对象的属性值不为object的时候,直接复制参数对象的每一个键值到新的对象对应的键值对中。result[key] = obj[key];}}// 返回拷贝完成后数据return result;}return obj;
}
// 使用
let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {let data1 = deepCopy(data)data1.name = '隔壁小花'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

通过遍历数据返回一个拷贝后船新的数据

缺点:据说如果数据深度 > 1000+ 会爆栈

3、lodash函数库

使用lodash函数库来进行深拷贝

html

<script src="https://cdn.jsdelivr.net/npm/lodash@4.17.21/lodash.min.js"></script>

js

let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {// 核心代码let data1 = _.cloneDeep(data)data1.name = '隔壁小花'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

4、$.extend

html

<script src="http://code.jquery.com/jquery-2.1.1.min.js"></script>

js

let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {// 核心代码let data1 = $.extend(true,{},data);data1.name = '隔壁老王'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

小结一下

推荐使用第一种或者第二种,后面两种需要通过引入外部库才行

其实还有其他的方法可以实现深拷贝,比如 Proxy 、树遍历等

那浅拷贝又是什么?

万物皆有对立面,有深便有浅

刚刚我们说深拷贝是将原对象的各项属性的“值”(数组的所有元素)拷贝过来,不是引用地址,那么浅拷贝就是 拷贝原对象的引用地址

对于浅拷贝而言,就是只拷贝对象的引用,而不深层次的拷贝对象的值,多个对象指向堆内存中的同一对象,任何一个修改都会使得所有对象的值修改,因为它们公用一条数据

找到前面的例子和图,这就是一个最典型的浅拷贝

let obj = {name: '严家辉',age: 18
}
let obj1 = obj
obj1.age = 24

这下明白了吗?

啥,还不明白 ,那我们再写俩栗子吧

浅拷贝的实现

1、for in

const deepCopy = obj => {let result = typeof obj.splice === "function" ? [] : {};if (obj && typeof obj === 'object') {for (let key in obj) {// 我们直接去掉递归,让第二层的数据直接赋值(引用地址)result[key] = obj[key];}return result;}return obj;
}
let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {let data1 = deepCopy(data);data1.name = '隔壁老王'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

我们直接使用深拷贝的第二个实现方式删除递归,让它直接赋值(引用地址),这就实现了一个浅拷贝

2、Object.assign

let data = {name: '严家辉',age: 18,other: {gender: "男"}
}
const a = () => {// 核心代码let data1 = Object.assign({},data);data1.name = '城中村村花'data1.other.gender = '女'console.log('a函数',data1)
}
const b = () => console.log('b函数',data) // 老严
a()
b()

小结一下

对于第一层的数据来说确实是拷贝成功了,但是在对象的属性是引用类型数据时我们还是拷贝的引用地址

总结

深浅拷贝的区别就是前者将原数据重新复制了一份然后新开了一个地址,后者则是将原数据的引用地址拷贝了一份而已

如有错误,望各位不吝赐教。

参考文档

https://www.jianshu.com/p/f4329eb1bace

https://segmentfault.com/a/1190000018592552?utm_source=sf-related

这篇关于浅谈深浅拷贝|手摸手带你入坑的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/360555

相关文章

浅谈mysql的sql_mode可能会限制你的查询

《浅谈mysql的sql_mode可能会限制你的查询》本文主要介绍了浅谈mysql的sql_mode可能会限制你的查询,这个问题主要说明的是,我们写的sql查询语句违背了聚合函数groupby的规则... 目录场景:问题描述原因分析:解决方案:第一种:修改后,只有当前生效,若是mysql服务重启,就会失效;

Spring核心思想之浅谈IoC容器与依赖倒置(DI)

《Spring核心思想之浅谈IoC容器与依赖倒置(DI)》文章介绍了Spring的IoC和DI机制,以及MyBatis的动态代理,通过注解和反射,Spring能够自动管理对象的创建和依赖注入,而MyB... 目录一、控制反转 IoC二、依赖倒置 DI1. 详细概念2. Spring 中 DI 的实现原理三、

Java中ArrayList的8种浅拷贝方式示例代码

《Java中ArrayList的8种浅拷贝方式示例代码》:本文主要介绍Java中ArrayList的8种浅拷贝方式的相关资料,讲解了Java中ArrayList的浅拷贝概念,并详细分享了八种实现浅... 目录引言什么是浅拷贝?ArrayList 浅拷贝的重要性方法一:使用构造函数方法二:使用 addAll(

浅谈主机加固,六种有效的主机加固方法

在数字化时代,数据的价值不言而喻,但随之而来的安全威胁也日益严峻。从勒索病毒到内部泄露,企业的数据安全面临着前所未有的挑战。为了应对这些挑战,一种全新的主机加固解决方案应运而生。 MCK主机加固解决方案,采用先进的安全容器中间件技术,构建起一套内核级的纵深立体防护体系。这一体系突破了传统安全防护的局限,即使在管理员权限被恶意利用的情况下,也能确保服务器的安全稳定运行。 普适主机加固措施:

浅谈PHP5中垃圾回收算法(Garbage Collection)的演化

前言 PHP是一门托管型语言,在PHP编程中程序员不需要手工处理内存资源的分配与释放(使用C编写PHP或Zend扩展除外),这就意味着PHP本身实现了垃圾回收机制(Garbage Collection)。现在如果去PHP官方网站(php.net)可以看到,目前PHP5的两个分支版本PHP5.2和PHP5.3是分别更新的,这是因为许多项目仍然使用5.2版本的PHP,而5.3版本对5.2并不是完

Linux 使用rsync拷贝文件

显示进度条 rsync 可以显示进度条,您可以使用 --progress 或 -P 选项来显示每个文件的传输进度和已完成文件的统计信息。 显示进度条的常用选项: --progress 选项 使用 --progress 显示每个文件的传输进度信息:rsync -av --progress /src/ /dest/ -a:归档模式,表示递归拷贝并保持文件权限、时间戳等。-v:详细模式,显示更

python基础语法十一-赋值、浅拷贝、深拷贝

书接上回: python基础语法一-基本数据类型 python基础语法二-多维数据类型 python基础语法三-类 python基础语法四-数据可视化 python基础语法五-函数 python基础语法六-正则匹配 python基础语法七-openpyxl操作Excel python基础语法八-异常 python基础语法九-多进程和多线程 python基础语法十-文件和目录操作

插件maven-search:Maven导入依赖时,使用插件maven-search拷贝需要的依赖的GAV

然后粘贴: <dependency>    <groupId>mysql</groupId>    <artifactId>mysql-connector-java</artifactId>    <version>8.0.26</version> </dependency>

浅谈java向上转型和乡下转型

首先学习每一种知识都需要弄明白这知识是用来干什么使用的 简单理解:当对象被创建时,它可以被传递给这些方法中的任何一个,这意味着它依次被向上转型为每一个接口,由于java中这个设计接口的模式,使得这项工作不需要程序员付出任何特别的努力。 向上转型的作用:1、为了能够向上转型为多个基类型(由此而带来的灵活性) 2、使用接口的第二个原因却是与使用抽象基类相同,防止客户端创建该类的对象,并确保这仅仅

JS手写实现深拷贝

手写深拷贝 一、通过JSON.stringify二、函数库lodash三、递归实现深拷贝基础递归升级版递归---解决环引用爆栈问题最终版递归---解决其余类型拷贝结果 一、通过JSON.stringify JSON.parse(JSON.stringify(obj))是比较常用的深拷贝方法之一 原理:利用JSON.stringify 将JavaScript对象序列化成为JSO