深拷贝

首先说一下深拷贝的实现对象肯定是引用类型,比如Array和Object这样的,因为基本类型比如Number、String的值不可变的,这里的不可变就是数值本身的不可变,比如说2你不能凭空变成1,如果你说我的数值变量能够多次赋值啊,那和我这个不一样,你的是基本数据类型在栈中存储,栈中存的值,然后你赋给另外的值,那就是栈里面存的值变成了新的赋值。
然后这里也顺便说,引用对象的变量是存在栈里面的,存的是被引用对象在堆里面的地址

  • 栈是自动分配固定大小的空间内存,比如int是多少字节,short又是多少字节,然后系统自动回收,先进后出🐴。
  • 然后堆是动态分配内存,内存大小不固定,也不会自动释放,堆数据结构是一种无序的树状结构,同时它还满足key-value键值对的存储方式;我们只用知道key名,就能通过key查找到对应的value

然后说道哪里了?哦继续讲深拷贝

var obj1={
a:1,
b:{
	c:1
}
var obj2=obj1
var obj3=shallowCopy(obj1)
var obj4=deepCopy(obj2)

现在上面有三种拷贝方法来拷贝obj1,分别是赋值、浅拷贝和深拷贝

  • 对于obj2来说,就相当于在栈中给了一个变量,存储obj1对象在堆内存的中存储地址,所以obj1和obj2实质上是相同的,两个是互相在操作同一个对象
  • 对于shallowCopy来说,字面意思,只是浅浅的复制一层,涉及深的地方,也就是引用对象的地方,像代码中的b对象,就还是和obj1的b对象是一样的,存了个对象的地址,对于obj3来说它可以自由操控浅浅的a,不递归的拷贝,不深层次遍历的拷贝
  • 对于deepCopy深拷贝来说,就是完全的拷贝,实际上是递归的拷贝,深层次的拷贝
  • 下面代码就是完整实现
  • 另外要注意的是在判断是否是引用类型的时候注意点,也就是怎么判断一个数据的类型[指路]
  • 在下面主要用了
let arr=[]
let obj={}
console.log(typeof arr)//object
console.log(typeof obj)//object
console.log(typeof arr == 'object') //true
console.log(typeof arr == 'array')//false
console.log(typeof obj == 'object')//true
console.log(arr.constructor == Object) //false
console.log(arr.constructor == Array)//true
console.log(obj.constructor == Object)//true
var a=[]
console.log(typeof a )//'object'
// console.log(typeof a == object)// ReferenceError: object is not defined
console.log(typeof a == 'object')// true
console.log(typeof a == 'array')// false
console.log( a.constructor=== Object)//false
console.log(a.constructor === Array)//true
//遍历对象
// var obj = {'0':'a','1':'b','2':'c'};
// Object.keys(obj).forEach(function(key){
//      console.log(key,obj[key]);
// });
// const object1 = {
//   a: 'somestring',
//   b: 42
// };
//用for of 遍历对象 
//注意和map set的差别 for (let [key, value] of mySet.entries()) 
// for (const [key, value] of Object.entries(object1)) {
//   console.log(`${key}: ${value}`);// a: 'somestring'
//   console.log(object1[key])//somestring
//   console.log(object1.key)//undefined 不能用.和变量来取属性值
// }

const shallowCopy=function(obj){
  var newCon={}
  for(let i in obj){
    newCon[i]=obj[i]
  }
  return newCon
}

const deepCopy=function(obj){
  let newCon=obj.constructor==Object?{}:[]
  if(typeof obj !== 'object'){
    return ;
  }else{
    // 第一种
  //  return  newCon=JSON.parse(JSON.stringify(obj))
  // 第二种
  // for(let i in obj){
  //   // console.log(i) //属性名
  //   // console.log(obj[i])//属性值
  //   // console.log(obj.i)//undefined
  //   if(obj[i].constructor==Object){
  //     newCon[i]=shallowCopy(obj[i])
  //   }else{
  //     newCon[i]=obj[i]
  //   }
  // }
  // return newCon
  // 第三种
  for(const [key,value] of Object.entries(obj)){
    if(typeof value == 'object'){
      newCon[key]=deepCopy(obj[key])
    }else {
      newCon[key]=value
    }
  }
    return newCon
  }
}
let obj1={
    a:1,
    b:{
      c:1
    }
}
let obj2=deepCopy(obj1)
let obj3=shallowCopy(obj1)

console.log('first:')
console.log(obj1)//{ a: 1, b: { c: 1 } }
console.log(obj2)//{ a: 1, b: { c: 1 } }
console.log(obj3)//{ a: 1, b: { c: 1 } }


obj2.a=2
obj2.b.c=2

obj3.a=3
console.log('second:')
console.log(obj1)//{ a: 1, b: { c: 1 } }
console.log(obj2)//{ a: 2, b: { c: 2 } }
console.log(obj3)//{ a: 3, b: { c: 1 } }

obj3.b.c=3
console.log('third:')
console.log(obj1)//{ a: 1, b: { c: 3 } }
console.log(obj2)//{ a: 2, b: { c: 2 } }
console.log(obj3)//{ a: 3, b: { c: 3 } }

好的👌🏻,去跳绳了跳绳🤸🏽♀️