16

前言

对于 数字boolean字符串 等基本类型 而言,赋值、浅拷贝和深拷贝无意义,因为每次都会在堆中开辟一块新的空间,指向新的地址。

一、赋值:

指向同一个地址,不拷贝。
赋值

var obj1 = {name:'圆', radius:10, point:{x:0,y:0}};
var obj2 = obj1;
 
obj2.name = "圆2";  //obj1中的name也会变

二、浅拷贝:

浅拷贝

var obj1 = {name:'圆', radius:10, point:{x:0,y:0}};
var obj2 = Object.assign({},obj1);
 
obj2.name="圆2"  // obj1.name不会变
obj2.point.x = 2       //obj1.point.x 改变,因为只拷贝到point一层

同样,解构赋值也是如此
var obj1 = {name:'圆', radius:10, point:{x:0,y:0}};
var obj2 = {…obj1}

除此之外: Array.prototype.concat()、Array.prototype.slice()、Array.from()和 Object.create()
都创建的是浅拷贝

所有 JavaScript 对象的标准内置复制操作都会创建浅拷贝,而不是深拷贝

三、深拷贝:

深拷贝

方法1

JSON.stringify(obj) 先将对象转换为字符串
JSON.parse(str) 然后再将字符串转为对象。
var obj1 = {name:'圆', radius:10, point:{x:0,y:0}};
var obj2 = JSON.stringify(obj1 );
var obj2 = JSON.parse(obj2);
 
obj2.name = "圆2";  // obj1 不变
obj2.point.x = 3;     //  obj1 不变

但JSON.stringify在转换Date,RegExp对象及function时会出现问题,同时也会忽略undefined、function

//date 类型
var o = new Date();
console.log(o.toString());         //  Mon Nov 06 2017 11:23:35 GMT+0800 (China Standard Time)  本地标准时间
console.log(JSON.stringify(o));    // "2017-11-06T03:23:35.547Z"  国际标准时间

因为stringify默认调用的是Object的toJSON方法,所以重写Date的toJSON,然后stringify就是ok的。

Date.prototype.toJSON = function () {
  return this.toLocaleString();
}
console.log(JSON.stringify(o));      // "11/6/2017, 11:23:35 AM"

同理RegExp

//RegExp类型
r1 = /\d+/;
console.log(JSON.stringify(r1));           //   {}
 
RegExp.prototype.toJSON = function(){
return this.toString();
}
console.log(JSON.stringify(r1));          //    "/\\d+/" 

此外window.structuredClone() api可上述效果一样,只能深拷贝可序列化的对象。

方法2

类库的方式。jquery,lodash等库
//jquery
let  y = $.extend(true,{},x)   //第一个参数 必须为true

//lodash库
let  y = _.cloneDeep(x);

寒水寺一禅
2.3k 声望119 粉丝

人生短短急个球!