上面代码中,源对象obj1
的a
属性的值是一个对象,Object.assign
拷贝得到的是这个对象的引用。这个对象的任何变化,都会反映到目标对象上面。
(2)同名属性的替换
对于这种嵌套的对象,一旦遇到同名属性,Object.assign
的处理方法是替换,而不是添加。
const target = { a: { b: ‘c’, d: ‘e’ } }
const source = { a: { b: ‘hello’ } }
Object.assign(target, source)
// { a: { b: ‘hello’ } }
上面代码中,target
对象的a
属性被source
对象的a
属性整个替换掉了,而不会得到{ a: { b: 'hello', d: 'e' } }
的结果。这通常不是开发者想要的,需要特别小心。
一些函数库提供 Object.assign
的定制版本(比如 Lodash 的_.defaultsDeep
方法),可以得到深拷贝的合并。
(3)数组的处理
Object.assign
可以用来处理数组,但是会把数组视为对象。
Object.assign([1, 2, 3], [4, 5])
// [4, 5, 3]
上面代码中,Object.assign
把数组视为属性名为 0、1、2 的对象,因此源数组的 0 号属性4
覆盖了目标数组的 0 号属性1
。
(4)取值函数的处理
Object.assign
只能进行值的复制,如果要复制的值是一个取值函数,那么将求值后再复制。
const source = {
get foo() { return 1 }
};
const target = {};
Object.assign(target, source)
// { foo: 1 }
上面代码中, source
对象的foo
属性是一个取值函数,Object.assign
不会复制这个取值函数,只会拿到值以后,将这个值复制过去。
常见用途
Object.assign
方法有很多用处。
(1)为对象添加属性
class Point {
constructor(x, y) {
Object.assign(this, {x, y});
}
}
上面方法通过Object.assign
方法,将x
属性和y
属性添加到Point
类的对象实例。
(2)为对象添加方法
Object.assign(SomeClass.prototype, {
someMethod(arg1, arg2) {
···
},
anotherMethod() {
···
}
});
// 等同于下面的写法
SomeClass.prototype.someMethod = function (arg1, arg2) {
···
};
SomeClass.prototype.anotherMethod = function () {
···
};
上面代码使用了对象属性的简洁表示法,直接将两个函数放在大括号中,再使用assign
方法添加到SomeClass.prototype
之中。
(3)克隆对象
function clone(origin) {
return Object.assign({}, origin);
}
上面代码将原始对象拷贝到一个空对象,就得到了原始对象的克隆。
不过,采用这种方法克隆,只能克隆原始对象自身的值,不能克隆它继承的值。如果想要保持继承链,可以采用下面的代码。
function clone(origin) {
let originProto = Object.getPrototypeOf(origin);
return Object.assign(Object.create(originProto), origin);
}
(4)合并多个对象
将多个对象合并到某个对象。
const merge = (target, …sources) => Object.assign(target, …sources);
如果希望合并后返回一个新对象,可以改写上面函数,对一个空对象合并。
const merge = (…sources) => Object.assign({}, …sources);
(5)为属性指定默认值
const DEFAULTS = {
logLevel: 0,
outputFormat: ‘html’
};
function processContent(options) {
options = Object.assign({}, DEFAULTS, options);
console.log(options);
// …
}
上面代码中,DEFAULTS
对象是默认值,options
对象是用户提供的参数。Object.assign
方法将DEFAULTS
和options
合并成一个新对象,如果两者有同名属性,则option
的属性值会覆盖DEFAULTS
的属性值。
注意,由于存在浅拷贝的问题,DEFAULTS
对象和options
对象的所有属性的值,最好都是简单类型,不要指向另一个对象。否则,DEFAULTS
对象的该属性很可能不起作用。
补充:
Object.assign()到底是浅拷贝还是深拷贝?
一、Object.assign()的用法:
1、Object.assign()方法的第一个参数是目标对象,后面的参数都是源对象,Object.assign()方法用于将所有可枚举属性的值从一个或多个源对象复制到目标对象。它将返回目标对象。
var obj = {}; var o1 = { a: 1 }; var o2 = { b: 2 }; var obj = Object.assign(obj, o1, o2); console.log(obj); // { a: 1, b: 2 }
注意:如果目标对象与源对象有同名属性,或多个源对象有同名属性,则后面的属性会覆盖前面的属性。
const target = { a: 1, b: 1 }; const source1 = { b: 2, c: 2 }; const source2 = { c: 3 }; Object.assign(target, source1, source2); console.log(target); // {a:1, b:2, c:3}
2、Vue中的使用技巧:
由于Object.assign()有上述特性,所以我们在Vue中可以这样使用:Vue组件可能会有这样的需求:在某种情况下,需要重置Vue组件的data数据。此时,我们可以通过this.$data
获取当前状态下的data,通过this.$options.data()
获取组件初始状态下的data。然后只要使用Object.assign(this.$data, this.$options.data())
就可以将当前状态的data重置为初始状态,非常方便。
二、Object.assign()到底是深拷贝还是浅拷贝?
1、通过上面我们已经对Object.assign()的用法有了一定的了解,那么Object.assign()到底是深拷贝还是浅拷贝呢?
-
对象的浅拷贝:浅拷贝是对象共用的一个内存地址,对象的变化相互影响。
-
对象的深拷贝:简单理解深拷贝是将对象放到新的内存中,两个对象的改变不会相互影响。
2、测试1:
let srcObj = {'name': 'lilei', 'age': '20'}; let copyObj2 = Object.assign({}, srcObj); console.log('srcObj', srcObj); //'name': 'lilei', 'age': '20' console.log('copyObj2', copyObj2); //'name': 'lilei', 'age': '20' srcObj.name="zhangsan"; console.log('srcObj', srcObj); //'name': 'zhangsan', 'age': '20' console.log('copyObj2', copyObj2); //'name': 'lilei', 'age': '20' copyObj2.age="10"; console.log('srcObj', srcObj); //'name': 'zhangsan', 'age': '20' console.log('copyObj2', copyObj2); //'name': 'lilei', 'age': '10'
从上面可以看出我们使用Object.assign()将srcobj的属性拷贝到了copyobj2中,而当我们修改srcobj(源对象)的属性或是copyobj2(目标对象)的属性时,变化的只有对象本身,这样看来是深拷贝啊,别着急下结论我们继续测试;
3、测试2:
let srcObj = {'name': 'lilei', 'grade': {'chi':"80", 'eng':"100"}}; let copyObj2 = Object.assign({}, srcObj); console.log('srcObj', srcObj); //name: "lilei" grade: {chi: "80", eng: "100"} console.log('copyObj2', copyObj2); //name: "lilei" grade: {chi: "80", eng: "100"}
还是和上面一样,我们先将srcobj对象的属性拷贝到目标对象中,正常打印输出;
1)首先我们在上面代码的基础上修改源对象的属性:
let srcObj = {'name': 'lilei', 'grade': {'chi':"80", 'eng':"100"}}; let copyObj2 = Object.assign({}, srcObj); srcObj.name="zhangsan"; srcObj.grade.chi="10"; console.log('srcObj', srcObj); //name: "zhangsan" grade: {chi: "10", eng: "100"} console.log('copyObj2', copyObj2); //name: "lilei" grade: {chi: "10", eng: "100"}
2)我们试试修改目标对象的属性,看看对源对象有什么影响:
let srcObj = {'name': 'lilei', 'grade': {'chi':"80", 'eng':"100"}}; let copyObj2 = Object.assign({}, srcObj); copyObj2.name="zhangsan"; copyObj2.grade.chi="50"; console.log('srcObj', srcObj); //name: "lisi" grade: {chi: "50", eng: "100"} console.log('copyObj2', copyObj2); //name: "zhangsan" grade: {chi: "50", eng: "100"}
分析:从例子中可以看出,当我们修改目标对象的属性值时,源对象的name没有变化,但是grade.chi却被改变了(修改源对象的属性也是同理),因此我们可以看出Object.assign()拷贝的只是属性值,假设源对象的属性值是一个指向对象的引用,那么它也只拷贝那个引用值。
总结:也就是说,对于Object.assign()而言,如果对象的属性值为简单类型(string,number),通过Object.assign({},srcobj);
得到的新对象为深拷贝;如果属性值为对象或其他引用类型,那对于这个对象而言其实是浅拷贝的,这是Object.assign()特别需要注意的地方。
另外多说一句,Object.assign({},src1,src2);
对于src1和src2之间相同的属性是直接覆盖的,如果属性值为对象,是不会对对象直接的属性进行合并的。
补充:Object.assign不会在那些源对象值为null或undefined的时候抛出错误。
自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。
深知大多数前端工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则几千的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!
因此收集整理了一份《2024年Web前端开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上前端开发知识点,真正体系化!
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)
最后
整理面试题,不是让大家去只刷面试题,而是熟悉目前实际面试中常见的考察方式和知识点,做到心中有数,也可以用来自查及完善知识体系。
《前端基础面试题》,《前端校招面试题精编解析大全》,《前端面试题宝典》,《前端面试题:常用算法》PDF完整版点击这里领取
由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!**
如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:前端)
最后
整理面试题,不是让大家去只刷面试题,而是熟悉目前实际面试中常见的考察方式和知识点,做到心中有数,也可以用来自查及完善知识体系。
《前端基础面试题》,《前端校招面试题精编解析大全》,《前端面试题宝典》,《前端面试题:常用算法》PDF完整版点击这里领取
[外链图片转存中…(img-6r88Bya1-1712877952756)]
[外链图片转存中…(img-df1ciJIU-1712877952757)]
[外链图片转存中…(img-LWgI5eN1-1712877952757)]