JS深拷贝函数 (开箱即用)

本文会讲到几种深拷贝的方法,

1.递归拷贝

这种方式 适用于多层数据嵌套

export function deepClone(obj){  //可传入对象 或 数组
 //  判断是否为 null 或 undefined 直接返回该值即可,
 if(obj === null || !obj)return obj; 
 // 判断 是要深拷贝 对象 还是 数组
 if(Object.prototype.toString.call(obj)==="[object Object]"){ //对象字符串化的值会为 "[object Object]"
   let target = {}; //生成新的一个对象
   const keys =  Object.keys(obj); //取出对象所有的key属性 返回数组 keys = [ ]
   //遍历复制值, 可用 for 循环代替性能较好
   keys.forEach(key=>{
     if(obj[key]&&typeof obj[key] === "object")
        //如果遇到的值又是 引用类型的 [ ] {} ,得继续深拷贝
       target[key] = deepClone(obj[key]);//递归
     else
       target[key] = obj[key];
     
   })
   return target  //返回新的对象
 }else if(Array.isArray(obj)){
   // 数组同理
   let arr = [];
   obj.forEach((item,index)=>{
     if(item&&typeof item === "object")
       arr[index] = deepClone(item);
     else
        arr[index] = item;
   })
   return arr
 }  
}

用法
const  obj = {};
const  obj2 = deepClone(obj);

总结:

  该方法比较好,对于一些复杂嵌套的数据类型可以完全深拷贝

.

2. JSON.stringify()

   const source = { name:'小明' ,sayName:()=>{} ,prop:undefined};
   const  newObj   = JSON.stringify( source )
  //先字符串序列化 再 parse解析回来
   console.log(JSON.parse(  newObj   ))
   VM575:1      {"name":"小明"}// 输出结果

总结:

  该方法简单,但是会吧引用类型和undefined的属性去除,相比第一种比较方便,但是比完全

.

3. ES6 扩展运算 ...

  //同样代码
   const source = { name:'小明' ,sayName:()=>{} ,prop:undefined, obj:{}};
   const  newObj   = {  ...source  }
  //先字符串序列化 再 parse解析回来
   console.log(  newObj  )
   VM575:1      {name: "小明", prop: undefined, sayName: ()=>{}, obj:{ }   }// 输出结果
    // 对二层结构的obj 赋值
    source.obj.name = 'rick'; 
    你会发现
    console.log( newObj.obj.name )    //  newObj.obj.name => rick 
    

总结:

  该方法简洁,比第二种方法好,但是,只是将 source 里面的值依次添加到 newObj,  
  当你把source.obj.name = 'rick';  newObj.obj.name 也等于 'rick'
  多层结构请不要用这个!!!!
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。