我是靠谱客的博主 满意毛衣,这篇文章主要介绍深入解析JavaScript中对象拷贝方法(附代码),现在分享给大家,希望可以做个参考。

说到javascript中的对象拷贝,首先我们想到的是Object.assign()

JSON.parse(JSON.stringify()),还有ES6的展开操作符[...]

因为在js=运算符 对于对象来说,不能创建副本,只是对该对象的引用

运算符

复制代码
1
2
3
4
5
6
7
8
var x = { a: 1, b: 2, }; y = x; x.a = 10; console.log(x); //{a:10, b:2} console.log(y); //{a:10, b:2}
登录后复制

所以在进行对象操作时,运算符等于号(=)不可取

Object.assign()

复制代码
1
2
3
4
5
6
7
8
var x = { a: 1, b: 2, }; y = Object.assign({}, x); x.a = 10; console.log(x); //{a:10, b:2} console.log(y); //{a:1, b:2}
登录后复制

初看,不会发现异常,因为所要的就是我们所要的结果,把对象结构弄再稍微复杂些再看

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
var x = { a: 1, b: 2, c: { d: 3, }, }; y = Object.assign({}, x); x.a = 5; console.log(x); //{a:5, b:2, c:{d:3}} console.log(y); //{a:5, b:2, c:{d:3}} x.c.d = 10; console.log(x); //{a:5, b:2, c:{d:10}} console.log(y); //{a:5, b:2, c:{d:10}}
登录后复制

此时就发现坑了,那么已经证明了Object.assign()只是实现了对象的浅拷贝

Object.assign()还需要注意的一点是,原型链上属性的不可枚举对象是无法复制的,看一下代码:

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
var x = { a: 1, }; var y = Object.create(x, { b: { value: 2, }, c: { value: 3, enumerable: true, }, }); var z = Object.assign({}, y); console.log(z); //{c:3}
登录后复制

拿到z的值很让人意外,因为xy的原型链,所以x不会被复制

属性b是不可枚举属性,也不会被复制

只有c具有可枚举描述,他可以被枚举,所以才能被复制

以上的坑也可以很好的被解决,且往下看:

深拷贝JSON.parse(JSON.stringify())

解决浅拷贝的坑

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
var x = { a: 1, b: 2, c: { d: 3, }, }; y = JSON.parse(JSON.stringify(x)); x.a = 5; x.c.d = 10; console.log(x); //{a:5, b:2, c:{d:10}} console.log(y); //{a:1, b:2, c:{d:3}}
登录后复制

当然普通的对象,此种复制方式已经是基本是完美了,那么他的坑在哪里呢

复制代码
1
2
3
4
5
6
7
8
9
10
11
var x = { a: 1, b: function b() { return "2"; }, }; y = JSON.parse(JSON.stringify(x)); z = Object.assign({}, x); console.log(y); //{a:1} console.log(z); //{a:1, b:function b(){return '2'}}
登录后复制

从结果看来,Object.assign()可以复制方法,JSON.parse(JSON.stringify())不可以

再来看第第二个坑:

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
var x = { a: 1, b: { c: 2, d: 3, }, }; x.c = x.b; x.d = x.a; x.b.c = x.c; x.b.d = x.d; var y = JSON.parse(JSON.stringify(x)); console.log(x); /* Uncaught TypeError: Converting circular structure to JSON at JSON.stringify (<anonymous>) at <anonymous>:8:25 */
登录后复制

报错了,其结果表明JSON.parse(JSON.stringify()),不能拷贝循环引用对象

再来看看Object.assign()

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
var x = { a: 1, b: { c: 2, d: 3, }, }; x.c = x.b; x.d = x.a; x.b.c = x.c; x.b.d = x.d; var y = Object.assign({}, x); console.log(x); /* [object Object]{ a:1, b:[object, Object], d:[object, Object], d:1 } */
登录后复制

使用展开操作符[... ]

对象字面量的展开操作符目前是ECMAScript的第 3 阶段提案,拷贝对象更加简单了

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
var x = [ "a", "b", "c", "d", { e: 1, }, ]; var y = [...x]; console.log(y); //['a', 'b', 'c', 'd', {'e':1}] var m = { a: 1, b: 2, c: ["d", "e"], }; var n = { ...m, }; console.log(n); //{a:1, b:2, c:['d', 'e']}
登录后复制

需要注意的是展开操作符也是浅拷贝。那么复制对象这厮真的这么难搞吗?

自己造轮子:

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
function copy(x) { var y = {}; for (m in x) { y[m] = x[m]; } return y; } var o = { a: 1, b: 2, c: { d: 3, e: 4, }, }; var p = copy(o);
登录后复制

有人说这么干应该没多大问题了吧。那么只能呵呵了,继续走

复制代码
1
2
3
4
5
6
7
8
9
10
var x = {}; Object.defineProperty(x, "m", { value: 5, writable: false, }); console.log(x.m); //5 x.m = 25; //这一步没报错,但是也没执行 console.log(x.m); //5
登录后复制

那么这样一来,展开操作符复制对象到这里也会遇到坑。

到处都是坑,防不胜防....我写到这估计还有许多坑都没完全列出来

之后再写吧

[完]

推荐学习:JavaScript视频教程

以上就是深入解析JavaScript中对象拷贝方法(附代码)的详细内容,更多请关注靠谱客其它相关文章!

最后

以上就是满意毛衣最近收集整理的关于深入解析JavaScript中对象拷贝方法(附代码)的全部内容,更多相关深入解析JavaScript中对象拷贝方法(附代码)内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(148)

评论列表共有 0 条评论

立即
投稿
返回
顶部