V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
southSu
V2EX  ›  分享发现

比 immutable 更简洁的 immutability-helper 不可变数据操作之使用总结

  •  
  •   southSu · 2018-11-13 12:54:32 +08:00 · 2758 次点击
    这是一个创建于 2253 天前的主题,其中的信息可能已经有所发展或是发生改变。

    immutability-helper 因 React 官方出镜,而被纳入后宫,首席填坑官∙苏南的专栏,梅斌的专栏

    引言

    之前项目中遇到数据拷贝、引用之间数据层级嵌套过深,拷贝的值相互之间影响的问题,后来引入了immutability-helper,使用过程中的一些总结,跟大家分享下,至于为什么不是immutable,请看下文分解,这里是@IT·平头哥联盟,我是首席填坑官——苏南

    作者:首席填坑官∙苏南 群:912594095,公众号:honeyBadger8,用心分享 做有温度的攻城狮。

    ​  相信大家在面试 /工作中都遇到过 js 对象 /数组的拷贝问题,面试官问你,你一般怎么做??在现在 ES6 盛行的当下,不会一点 ES6 都不好意思说自己是前端(其实我一般都说自己是攻城狮、切图崽),我们想的大多第一想法,如下:

    • Object.assign - 最方便;
    • [...] - 最有逼格;
    • JSON.parseJSON.stringify - 完美组合;
    • $.extend() - jQuery 时代的引领潮流时尚前沿的 API;
    • 最后想到的才是自己递归实现一个;

    但是通常我们使用的Object.assign属于浅拷贝,当数据嵌套层级较深时,就……呵呵了;而JSON.parse、stringify它应该是创建一个临时可能很大的字符串,然后又访问解析器,性能是比较慢的。于是后来发现了 immutable「不可变数据」,曾经我也一度特别喜欢它,但时间久了,慢慢发现,它过于有个性了些、凡事都都没有任何商量的余地,所有的数据,从创建、变更、插入、删除等操作,都要按它的套路来,对于我这种一生放荡不羁爱自由的人来说,长时间的约束,是不能忍的;都说两人如果三观不合,是无法长久下去的,可能也是缘份吧,在后来的某一天偶然的闲逛中邂逅了新欢 ————Immutability Helpers

    嗯,今天的主题就是给大家分享一下,Immutability Helpers的一些用法,会介绍 API 的使用操作和小技巧,如有不理解不对,请纠正:

    太兴奋了,差点忘了,补充一下,一个简单的拷贝:

      //实现一个简单的递归数据拷贝
      let customClone = (rawObj)=>{
        let copyObj = {};
    
        for (var key in rawObj) {
          if( typeof rawObj[key] === 'object' && Object.prototype.toString.call(rawObj[key]) !== '[object Array]'){
              copyObj[key] = customClone(rawObj[key]);
          }else{
              copyObj[key] = rawObj[key];,//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
          };
        };
        return copyObj;
      };
      let objA =  {"name":"苏南","sex":"男","height":"176"};
      let objB =  customClone(objA);
          objB.signature = "宝剑锋从磨砺出,梅花香自苦寒来,做有温度的攻城狮";
    
      console.log(objA);
      console.log(objB);
    
    

    展示 Object.assign 拷贝问题

    • 补充一个 Object.assign 的坑 :
      let data = {
        a:1,
        b:2,
        children:{
          name:"苏南",
          organization:"@IT·平头哥联盟",
          job:"首席填坑官",
          address:"ShenZhen",
          age:18
        }
      };
      let data2 = Object.assign({},data);
      data2.children.age = 28;
      data2.children.job = "首席甩锅官";
      data2.b = 666;
      console.log("我是原始数据 data:",data);
      console.log("我是复制后的数据 data2:",data2);
    
    

    展示 Object.assign 拷贝问题,公众号:honeyBadger8

    immutable 最后的一次回顾

    都说有了新欢,忘了旧爱,但我不是那种无情无义的人,最后正式介绍一下 immutable,为我俩的……画上一个圆满的句号:

    再次强调,并不是觉得immutable不好,不够强大,只是自己个人观点,有些不喜欢而已,各位immutable粉勿喷,想了解更多的同学可以点击这里

    Immutable data encourages pure functions (data-in, data-out) and lends itself to much simpler application development and enabling techniques from functional programming such as lazy evaluation.

    使用示例:
      
      const list1 = List([ 1, 2, 3 ]);
      const list2 = List([ 4, 5, 6 ]);
      const array = [ 7, 8, 9 ];
      const list3 = list1.concat(list2, array);
      console.log(list3) // List {size: 9, _origin: 0, _capacity: 9, _level: 5, _root: null, …} 是不能直接获取到数据的,须使用 get,-- list3.get(0)
    
    
    
      
      let data = fromJS({
        obj:{}
      });
      let data1 = {
        a:1,
        b:2,
        children:{
          name:"苏南",
        }
      };
      let data2 = data.mergeIn(['obj'],data1,{c:666});
      console.log("获取的数据:",data2.getIn(['obj','c']));
      console.log("这里是由 formJS 创建的数据:",data2.getIn(['obj','children','name']));//
    
    

    immtublejs 使用示例,简单演示

    使用 immutable 后,所有数据都要类似选择器,一个一个往下选择,并不是说它不好、功能不够强大,只是自己有些不喜欢它类似 JQuery 选择器一样的语法,get、getIn、set、List 等的使用方式,当然它也是可以使用 toJS方法转回来的。

    Immutability Helpers 出场

    gitHub 上它对自己的介绍很简单:Mutate a copy of data without changing the original source —— 在不更改原始源的情况下改变数据副本。

    与它结缘,是因为它在 react 官方文档中出镜,而被我所宠幸,真的 ,只是因为在人群中多看了它一眼再也没能忘掉, 它跟immutable不一样,不会有那么多条条框框约束你,给你自由、给你独立的空间、给你独立的思想,让你想用即用、用之即走~~(泥马,怎么有点像张小龙说它的小程序一样),但您放心,它的坑真的比小程序少,API 也很简洁,接下来来看一下,它的基本用法:

    • $push —— 数组;
    • $unshift —— 数组;
    • $splice —— 数组;
    • $set —— 替换 /覆盖 /合并原数据;
    • $toggle —— array of strings,toggles a list of boolean fields from the target object ;
    • $unset —— remove the list of keys in array from the target object;
    • $merge —— 合并对象;
    • $apply —— passes in the current value to the function and updates it with the new returned value ;
    • $add —— 新增;
    • $remove —— 删除。

    以上基本就是它全部的 API 了,下面一起来看看,具体用法吧:

    $push 的使用 :

    • 看名字就知道它的作用了啦,跟原生的push一样,不过写法有一点点不一样;
      let arr = [1,2,3,4,5,66];
      let arr2 = update(arr,{
        $push : ["a","b","c"], //一定要 []号的形式哦,不可以 "a";
        [4]:{ // !! index ,可以指定修改下标的值
          $set:"我是首席填坑官"
        }
      });
      console.log(arr2);
    

    $unshift 的使用 :

    • 一样,跟原生的unshift,在原数组开头处插入,同样写法是以一个数组的形式;
      let arr = [1,2,3,4,5,66];
      let arr2 = update(arr,{
        $unshift : ["a","b","c"],//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
        [4]:{
          $set:"我是首席填坑官∙苏南"  //这里需要注意,它的操作是在 unshift 之前执行的,也就是在原 arr 上查找 第 4 个下标
        }
      });
      console.log("原始数组",arr);// [1, 2, 3, 4, 5, 66] 相互之间并不会影响
      console.log(arr2); //["a", "b", "c", 1, 2, 3, 4, "我是首席填坑官∙苏南", 66]
    

    $splice 的使用 :

    • 注意 :数组套数组,start,end, 插入的数据……,;
    
      let arr = [1,2,3,4,5,66];
      let arr2 = update(arr,{
        $splice : [[1,2,[66788,99],{a:123,b:"苏南"}]], // or [0,1,"从我开始是插入的内容",88,89,90,"后面可以很多,是数组、对象、字符串都行"]
      });
      console.log(arr2); 
    
      //复杂一些的用法:
      let obj={
        name:"immutable",
        list :[1,2,[90,55,44,3,22,55],3,4,6,7,8]
      };
      let obj2 = update(obj,{
        list:{
          [2]:value=>update(value,{
            $splice:[[0,2]]  // [90,55,44,3,22,55] => [44, 3, 22, 55]
          })
        }
      });
    
    

    immutability-helper $splice 的使用展示

     $splice 的使用展示,格式错误的警告

    $set 的使用 :

    • 上面已经演示过了,其实有点替换的意思,当有重复的值时,就会覆盖,没有就新增,来展示复杂一点的场景,层级深的数据,也不会相互影响;
      let obj={
        name:"immutable",
        children:{
          address:"ShenZhen",
          hobby:"@IT·平头哥联盟-前端开发"
        }//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
      };
      let obj2 = update(obj,{
        $set : {name:"immutability-helper",other:"其他字段,如微信公众号:honeyBadger8,每周为你带来最新分享"}
      });
      let obj3 = update(obj,{
        name:{
          $set : "苏南"
        },
        children:{
          hobby:{
            $set:"首席填坑官 - javascript"
          }
        }
      });
      console.log("原始数据:",obj); 
      console.log("obj2:",obj2); 
      console.log("obj3",obj3); 
    

    immutability-helper $set 的使用展示

    $toggle 的使用:

    • 听名字,应该就能猜出来,开关切换的意思;
    • Boolean 布尔值的切换,如果你是强制要 Number 类型 的 0、1,那么使用引方法的时候就要注意了;
      let obj={
        name:"immutable",
        a:false,
        b:true,
        c:1,
        d:0
      };
      let obj2 = update(obj,{
        $toggle:['b','a',"c","d"],
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
    
    

    immutability-helper $toggle 的使用展示

    $unset 的使用:

    • 它跟$set相反,有点 remove 的味道,但又貌似有不同的之处,当操作的对象为object时 key 是删除了;而数组array中它的值没有了,却保留了下标,不改变数组的长度,删除数组建议还是用$splice;请看下图:
      let arr = [1,2,3,4,5,6];
      let obj={
        name:"immutable",,//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
        children:{
          address:"ShenZhen",
          hobby:"写博客"
        }
      };
      let obj2 = update(obj,{
        $unset : ["name"],//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
        children:{
          $unset:["address"]
        }
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
    
      let arr2 = update(arr,{
        $unset : [1]
      });
      console.log("arr2:",arr2,arr2.length);
    
    

    immutability-helper $unset 的使用展示

    $merge 的使用:

    • $merge 跟我们最爱的Object.assign一样,做合并操作的,但它比assign优秀很多,深层次拷贝,不会相互影响 :
      let arr = [1,2,3,4,5,6];
      let obj={
        name:"immutable",
        children:{
          address:"ShenZhen",
          hobby:"写博客",
          array:["我不是程序员","切图崽了解一下"],
        }
      };
      let obj2 = update(obj,{
        $merge:{
          arr
        },
        children:{
          array:{
            $merge:{items:["从前有坐山","山里有个庙"]},
            $splice:[[3,0,"住着一个小和尚"]]
          }
        }
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
    
    

    $merge 的使用展示,做有温度的攻城狮

    $apply 的使用:

    • $apply 基于当前值进行一个函数运算,从而得到新的值 :
    • 注意 :它必须是一个 function 哦!
      let obj={
        name:"immutable",
        children:{
          items:["从前有一坐山"],
          array: [1,2,3,4,5,6],
        }
      };
      let obj2 = update(obj,{
        name:{
          $apply:(val)=>("首席填坑官")
        },
        children:{
          items:{
            $apply:(val)=>{
              console.log("旧值",val);
              return [3,0,"住着一个小和尚"]
            }
          },
          array:{//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
            $apply:(val)=>(val.reverse()) //必须是一个函数
          }
        }
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
    
    

    $apply 的使用展示

    $apply 的使用展示,必须是 function 的错误警告

    $remove 的使用:

    • $remove 一定一定 要是使用SetMap 创建的数组:
    • 要删除的值,必须是数组成存在的,如值不存在则忽略,$remove:[2,666],2 会删除,6 则会被忽略;
    • 这个 api 有点奇怪,正常普通的数组 [],这样的删除不了!!;
    • 常见错误如下图:
      let obj={
        name:"immutable",
        children:{
          array:new Set([1, 2, 3, 4, 4]),
        }
      };
      let obj2 = update(obj,{
        children:{
          array:{
            $remove:[2],
          },
        }
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
    

    $remove 的使用展示,必须是 new Set Map 创建

    必须是 new Set Map 创建的报错

    $add 的使用:

    • $add 跟刚才的 $remove 一样要使用 Map/Set,$add 方法也跟 es6 Map/Set的 add 方法一致:
    • 只是写的时候也要注意一些, [ [] ] ,嵌套!
      let obj={
        name:"immutable",
        array:new Map([["a",1],["b",2]]),
      };
      let obj2 = update(obj,{
        array:{
          $add:[["66",56]],
        },//首席填坑官∙苏南的专栏,交流:912594095、公众号:honeyBadger8
      });
      console.log("原始数据:",obj);
      console.log("obj2:",obj2);
      console.log("获取 key a:",obj2.array.get('a'));
    
    

    本文由 @IT·平头哥联盟-首席填坑官∙苏南 分享

    Immutability Helpers 的高阶用法:

    • 还可以自定义方法,如 定义一个 $trinocular 方法,来判断数组中的值;
    • 只是一个简单的示例,更多复杂的用法,可以自己去探索哦 去官方 github
      update.extend('$trinocular', function(proportion, original) {
        return  original > 88 ? (original/proportion ): (proportion+original);
      });
      let array =[56,33,55,777,322,444,61,12,34,52,245];
      let array2 = array.map((k,v)=>update(k,{
        $trinocular:2
      }))
      console.log("原始数据:",array);
      console.log("array2:",array2);
    

    immutability-helper 高阶用法

    总结 /结尾:

    以上就是基础 API 的用法 ,添加了一些官方示例没有讲到的组合使用,以及使用过程中,可能出现的一些错误,需要留意的地方,更多定制高级用法,有兴趣的同学可以自行了解一下。

    以上就是今天为大家带来的分享,它可能没有 immutable 那么多功能,但贵在简洁,不会有太多的约束,如理解有误之处,欢迎各位大佬纠正,毕竟我还只是个宝宝——新手上路中!。

    下方是我弄的一个公众号,欢迎关注,以后文章会第一时间,在公众号上更新,原因是之前分享的有两篇文章,竟然被其他公众号抄袭了,前些天去更新发表的时候,微信提示我文章已经不是原创了检测到相同的文章,宝宝心里那个凉啊~,果断申诉告了对方(是一个培训学校公众号,好气哦),补了简书发布的链接和截图日期,万幸最后胜诉了!

    宝剑锋从磨砺出,梅花香自苦寒来,做有温度的攻城狮!

    更多文章:

    浅谈 easy-mock 最好的备胎没有之一 如何给 localStorage 设置一个过期时间? immutability 因 React 官方出镜之使用总结分享! 小程序项目之做完项目老板给我加了 6k 薪资~ 小程序项目之填坑小记 面试踩过的坑,都在这里了~ 你应该做的前端性能优化之总结大全! 动画一点点 - 如何用 CSS3 画出懂你的 3D 魔方? 动画一点点 - 手把手教你如何绘制一辆会跑车 SVG Sprites Icon 的使用技巧

    作者:苏南 - 首席填坑官 链接: https://blog.csdn.net/weixin_43254766/article/details/83472544 交流:912594095、公众号:honeyBadger8 本文原创,著作权归作者所有。商业转载请联系@IT·平头哥联盟获得授权,非商业转载请注明原链接及出处。

    3 条回复    2018-11-14 14:51:36 +08:00
    southSu
        1
    southSu  
    OP
       2018-11-13 12:55:38 +08:00
    一场红尘恋,
    一份千年缘,
    几缕隔岸相思,
    隐逸了多少楼台旧梦?——中午好,我是苏南,感谢您的耐心阅读与认可,欢迎指点!
    kernel
        2
    kernel  
       2018-11-13 13:35:27 +08:00
    何不用 immer
    xmsz
        3
    xmsz  
       2018-11-14 14:51:36 +08:00
    想知道实测 JSON.parse、stringify 的效率情况
    目前还是用这个方式,快捷又是原生
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1024 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 28ms · UTC 22:36 · PVG 06:36 · LAX 14:36 · JFK 17:36
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.