龙空技术网

深入理解js对象的引用

前端小白说 275

前言:

当前看官们对“js对象赋值是引用”大约比较关怀,我们都需要学习一些“js对象赋值是引用”的相关内容。那么小编在网上收集了一些有关“js对象赋值是引用””的相关知识,希望各位老铁们能喜欢,小伙伴们一起来学习一下吧!

JavaScript 有七种内置类型,其中:

基本类型

• 空值(null)• 未定义(undefined)• 布尔值( boolean)• 数字(number)• 字符串(string)• 符号(symbol,ES6 中新增)

引用类型

• 对象(object)

对于基本类型,赋值(=)是值的拷贝,比较(===)的是实际的值,而对于引用类型(Array也是一种Object),赋值(=)是引用地址的拷贝,比较(===)的是引用地址:

注:下面图例中用类似于X000001,X000002表示引用地址,只为了更好的举例说明,并不是真实的值

案例一

const a = '哈哈'const b = '哈哈'console.log(a === b) // trueconst c = {}const d = {}console.log(c === d) // false

注解:

1.a和b是字符串,比较的是值,完全相等2.c和d是对象,比较的是引用地址,c和d都是一个新对象,方别指向不同的地址,所以不相等

案例二

let a = { z: 5, y: 9 }let b = ab.z = 6delete b.yb.x = 8 console.log(a) // {z: 6, x: 8}console.log(a === b) // true

注解:

1.a是对象,b=a是将a的引用地址赋值给b2.a和b都指向与同一个对象,修改这个对象,a和b都会变化

案例三

let a = { z: 5 }let b = ab = {z: 6}console.log(a.z) // 5console.log(a === b) // false

注解:

1.a是对象,b=a是将a的引用地址赋值给b2.b = {z: 6}新对象赋值给b,切断了a和b的联系,分别指向于不同的对象

总结:(精髓所在)

只操作(修改,删除,添加)对象的属性,不会与之前对象断开连接(案例二)

直接操作对象本身,也就是最外层,会和之前的对象断开连接(案例三)

数组也是对象

案例四

let a = { z: 5, y: {x: 8}, w: {r: 10} }let b = {...a}b.z = 6b.y.x = 9b.w = {r: 11}console.log(a) // { z: 5, y: {x: 9}, w: {r: 10}}console.log(a.y === b.y) // trueconsole.log(a.w === b.w) // falseconsole.log(a === b) // false

注解:

1.b = {...a}中,z是基本类型直接拷贝值,y和w是对象,是引用地址的拷贝2.y是只操作属性,连接不会断开,w操作了本身,生产了一个新对象,连接断开(参考上面的总结)

案例四理解之后应该就知道为什么js对象有浅拷贝和深拷贝的区分了

应用

场景:目前有多个用户,每个用户有自己的属性,展示并且可以修改

程序实现(例如vue)

首先我们将每一个用户都封装成一个单独的模块(.vue),用户初始数据存放在model里面(vuex)

一般来说修改model里面数据,都需要用它的mutations和actions里面的方式,如果用户属性比较多,每一项都需要去定义一个mutations或actions的话,那开发量是相当的大

利用对象的引用关系,传过来的数据不和源对象,切断关系,是直接可以操作源对象,组件与组件之间的通信也可以这个实现

有利也有弊,这种操作起来很简单,但一旦切断他们的联系之后,不好维护,用这种方式需要对对象引用有深入的理解,知道什么时候会断开联系

个人建议只在这种多个相同组件中使用。

标签: #js对象赋值是引用 #js对象数组赋值引用地址问题