Javascript 浅拷贝与深拷贝
在了解JS的浅拷贝与深拷贝之前,我们需要先知道什么是值传递与引用传递。
在JS中,基本类型值的拷贝是按值传递的,而引用类型值的拷贝则是按引用传递的。通过值传递的变量间不会有任何牵连,互相独立;但是引用传递拷贝的变量间则会相互影响,修改其中任何一方所引用的对象的值都会在另一方中体现,之所以会有这样的表现和JS的内存机制有关。
JS的内存也分为堆和栈,但是注意这里的堆栈和数据结构的堆栈是不同的概念。
栈:由系统自动分配,自动回收,效率高,但容量小
堆:由程序员手动分配内存,并且手动销毁(高级语言如JS中有垃圾自动回收机制),效率不如栈,但容量大
JS定义的基本类型值会被存放在栈内存中,JS可以直接访问栈内存,所以访问存放基本类型值的变量可以直接访问到基本类型值。而引用类型因为其大小不固定,系统会为引用类型分配堆内存存放,而只将指向该堆内存空间的指针(引用)存放在栈中,JS不予许直接访问存放引用类型值的堆内存,只能通过访问引用间接的访问引用类型值。这样一来,我们访问引用类型值时,实质上只是在访问它的引用,然后再按照这个引用地址去堆中找到它的实际内容。
因此进行拷贝的时候,对于基本类型值变量,系统会为新变量单独开辟一个新的栈内存空间,并将源变量保存的基本类型值拷贝一份保存到里面。而对于引用类型值,新变量拷贝得到的只是引用对象的引用而已,这么一来,通过两个变量保存的引用访问到的实质就是同一块堆内存,也就是同一个对象。
let person = {
name : "kelly",
love : { sport : "swim", movie : "love story" }
}
let new_person = person
new_person.name = "lin"
console.log( person )
console.log( new_person )
浅拷贝是指在进行对象拷贝的时候,只对对象的第一层键值进行拷贝。
在上面的例子当中,如果不希望修改 new_person 对象的 name 值的时候,源对象的 name 值也跟着一起改变,我们可以尝试对拷贝过程做一些处理,而不再只是简单的直接赋值。
let person = {
name : "kelly",
love : { sport : "swim", movie : "love story" }
}
let new_person = { } function shallowCopy( target, source ){
if( !source || typeof( source ) !== "object" ){
return
}
if( !target || typeof( target ) !== "object" ){
return
}
for( let key in source ){
if( source.hasOwnProperty( key ) ){
target[ key ] = source[ key ]
}
}
} shallowCopy( new_person, person )
new_person.name = "lin"
console.log( new_person )
console.log( person )
这时候我们修改了拷贝对象的 name 值,源对象的 name 值不会再跟着改变了,可是当我们修改属性 sport 的值的时候,源对象的 sport 值却又跟着改变了。
shallowCopy( new_person, person )
new_person.love.sport = "run"
console.log( new_person )
console.log( person )
Object.assign( new_person, person )
new_person.name = "lin"
new_person.love.sport = "run"
console.log( new_person )
console.log( person )
除此之外,Array 的 concat 方法和 slice 方法实现的也是数组对象的浅拷贝。
let arr = [ , [ , , ] ]
let arr_concat = arr.concat( )
let arr_slice = arr.slice( )
arr_concat[ ][ ] = "NO"
console.log( arr ) //[ 1, [ 'NO', 3, 4 ] ]
console.log( arr_concat ) //[ 1, [ 'NO', 3, 4 ] ]
arr_slice[ ][ ] = "NONO!"
console.log( arr ) //[ 1, [ 'NO', 'NONO!', 4 ] ]
console.log( arr_slice ) //[ 1, [ 'NO', 'NONO!', 4 ] ]
数组 arr,arr_concat, arr_slice 的第二个元素引用的都是同一个堆内存对象。
function deepCopy( target, source ){
if( !source || typeof( source ) !== "object" ){
return
}
if( !target || typeof( target ) !== "object" ){
return
}
for( let key in source ){
if( source.hasOwnProperty( key ) ){
if( source[ key ] && typeof( source[ key ] ) === "object" ){
target[ key ] = target.constructor === "Array" ? [ ] : { }
deepCopy( target[ key ], source[ key ] )
} else {
target[ key ] = source[ key ]
}
}
}
} deepCopy( new_person, person )
new_person.name = "lin"
new_person.love.sport = "run"
console.log( new_person )
console.log( person )
var obj = {
a: 1,
b: 2,
c: undefined,
sum: function() { return a + b }
}; var obj2 = JSON.parse( JSON.stringify( obj ) )
console.log( obj2 )
Javascript 浅拷贝与深拷贝的更多相关文章
- JS面试题-<变量和类型>-JavaScript浅拷贝与深拷贝
前言 最开始了解到深浅拷贝是因为准备面试,但那个时候因为在学校做的项目比较少需求也比较简单,所以没有在项目中遇到这类问题,所以对这个问题就属于知道这个知识点,看过相关内容,却没有自己的总结,也没有深入 ...
- 浅谈Javascript 浅拷贝和深拷贝的理解
javascript中存储对象都是存地址的. 浅拷贝:浅拷贝是都指向同一块内存区块,浅拷贝共用同一内存地址,你改值我也变.如果拷贝的对象里面的值是一个对象或者数组,它就是浅拷贝,拷贝的知识引用地址. ...
- javascript浅拷贝和深拷贝
/* 浅拷贝 */ function extend(parent, child) { var i; child = child || {}; for (i in parent) { if (paren ...
- 关于JavaScript的浅拷贝和深拷贝
在 JS 中有一些基本类型像是Number.String.Boolean,而对象就是像这样的东西{ name: 'Larry', skill: 'Node.js' },对象跟基本类型最大的不同就在于他 ...
- Javascript中的浅拷贝和深拷贝
很多开发语言中都有浅拷贝和深拷贝的说法,这里简单区分一下它们在Javascript中的区别,以及jQuery中深拷贝的实现. 在谈浅拷贝和深拷贝之前,先要屡清楚Javascript中的按值访问和按引用 ...
- javascript篇-浅拷贝与深拷贝
理解javascript 的浅拷贝与深拷贝,首先看一下js的数据类型: js有5种基本数据类型:undefined,null,boolean,number,string 还有一种复杂的数据类型(也叫引 ...
- javascript浅拷贝深拷贝理解记录
javascript的深拷贝和浅拷贝问题几乎是面试必问的问题.好记性不如烂笔头,特此来记录一下自己对深拷贝浅拷贝的理解. 顾名思义,拷贝就是copy复制,在js中可以浅而理解为对一个对象或者数组的复制 ...
- javascript中的浅拷贝和深拷贝(拷贝引用和拷贝实例)
作者:千锋教育链接:https://www.zhihu.com/question/23031215/answer/326129003来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载请 ...
- JavaScript 数据结构与算法之美 - 栈内存与堆内存 、浅拷贝与深拷贝
前言 想写好前端,先练好内功. 栈内存与堆内存 .浅拷贝与深拷贝,可以说是前端程序员的内功,要知其然,知其所以然. 笔者写的 JavaScript 数据结构与算法之美 系列用的语言是 JavaScri ...
随机推荐
- 搭建基于hyperledger fabric的联盟社区(七) --升级chaincode
上个版本的chaincode有很多功能不完备,所以要部署新版本的chaincode.Fabric支持在保留现有状态的前提对chaincode进行升级. 一.新版chaincode 新版本的chainc ...
- Gradle: Can't load library: native-platform.dll
Eclipse 导入 Gradle project 时总是报错:Can't load library: native-platform.dll. 解决方案: 进入 Windows -> Pref ...
- postman 设置代理
点击右上角 图标(亮着的为录制中) 设置端口 和存放位置 把浏览器设置代理 localhost 8080 即可 filter中可以通过正则表达式来匹配自己关心的url 2018.9 后记: 今 ...
- 杂项:大数据 (巨量数据集合(IT行业术语))
ylbtech-杂项:大数据 (巨量数据集合(IT行业术语)) 大数据(big data),指无法在一定时间范围内用常规软件工具进行捕捉.管理和处理的数据集合,是需要新处理模式才能具有更强的决策力.洞 ...
- python调试方法
之前调试python程序都是用print参数,感觉有点弱爆啊,最近发现python也有类似C语言gdb的工具pdb,记录下pdb的使用方法和心得. 先找了段简单的测试程序: 复制代码 !/usr/bi ...
- html2pdf 中文支持问题
系统用的是HTML2PDF V4.0.3 版本 百度后 http://blog.sina.com.cn/s/blog_6b0ce0310101fdv6.html 发现中文支持不好 还是有乱码问题 解决 ...
- Oracle播放多条 INSERT ALL
Oracle INSERT ALL 语句介绍 Oracle INSERT ALL 语句用来用一个 INSERT 语句添加多行.该行可以只使用一个SQL命令插入到一个表或多个表. 语法 Oracle I ...
- Eclipse的基本使用
01Eclipse的下载安装 * A: Eclipse的下载安装 * a: 下载 * http://www.eclipse.org * b: 安装 * 只需要解压后就能使用 * c: 卸载 * 只 ...
- windows下使用SQLPLUS制作BAT执行SQL文件
假如你把需要的SQL操作信息等均放入到一个SQL文件中,需要制作一个bat文件来执行这个sql文件,那么你的bat文件中,在sqlplus登录语句后的信息不能换行,换行的话则执行登录sqlplus后就 ...
- 表单验证常用的JS正则表达式
在表单验证中,使用正则表达式来验证正确与否是一个很频繁的操作,本文收集整理了15个常用的javaScript正则表达式,其中包括用户名.密码强度.整数.数字.电子邮件地址(Email).手机号码.身份 ...