深入解析JavaScript中对象拷贝方法(附代码)

之前的文章《你值得了解的JavaScript“继承之jquery”使用方法(代码详解)》中,给大家了解了JavaScript“继承之jquery”使用方法。下面本篇文章给大家了解一下JS中对象拷贝方法,有需要的朋友可以参考一下。

深入解析JavaScript中对象拷贝方法(附代码)

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

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

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

运算符

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}

登录后复制

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

立即学习“Java免费学习笔记(深入)”;

Object.assign()

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}

登录后复制

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

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()还需要注意的一点是,原型链上属性的不可枚举对象是无法复制的,看一下代码:

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的值很让人意外,因为x是y的原型链,所以x不会被复制

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

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

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

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

解决浅拷贝的坑

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}}

登录后复制

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

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())不可以

再来看第第二个坑:

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 ()    at :8:25*/

登录后复制

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

再来看看Object.assign()

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 阶段提案,拷贝对象更加简单了

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']}

登录后复制

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

自己造轮子:

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);

登录后复制

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

var x = {};Object.defineProperty(x, "m", {  value: 5,  writable: false,});console.log(x.m); //5x.m = 25; //这一步没报错,但是也没执行console.log(x.m); //5

登录后复制

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

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

之后再写吧

[完]

推荐学习:JavaScript视频教程

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2708648.html

(0)
上一篇 2025年3月7日 20:24:09
下一篇 2025年2月27日 23:09:03

AD推荐 黄金广告位招租... 更多推荐

相关推荐

发表回复

登录后才能评论