怎么用redux实现computed计算属性-mile米乐体育
怎么用redux实现computed计算属性
这篇文章主要介绍“怎么用redux实现computed计算属性”,在日常操作中,相信很多人在怎么用redux实现computed计算属性问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么用redux实现computed计算属性”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
前言
什么是computed计算属性?它会根据所依赖的数据动态显示新的计算结果, 该计算结果会被缓存起来。如果是vue开发者,对这个功能并不陌生,而且很常用。对于react开发者,如果用过mobx,那其实也不陌生,一个装饰器就生效了????。那如果是redux呢??(沉默中。。。)有了,reselect嘛,哈哈????。啪,骗子,这是假的计算属性,它要手动提供全部依赖,每个依赖都是一个函数回调确定依赖值,每次写这么多代码是有多想敲坏我的机械键盘(嘶吼)。
这么说,redux和计算属性无缘?也不能这么说,办法总比困难多。虽然redux是单向数据流,无法做响应式操作,不过,我们可以创造出一个监听对象
import{store}from'redux';constcollector=[];classobjectdeps{protectedreadonlydeps:string[];protectedreadonlyname:string;protectedreadonlystore:store;protectedsnapshot:any;constructor(store:store,name:string,deps:string[]=[]){this.store=store;this.name=name;this.deps=deps;collector.push(this);}proxy(currentstate){if(state===null||typeofstate!='object')returnstate;constproxydata=array.isarray(state):[]:{};constcurrentdeps=this.deps.slice();constkeys=object.keys(currentstate);for(leti=keys.length;i-->0;){constkey=keys[i]!;object.defineproperty(proxydata,key,{enumerable:true,get:()=>{if(visited){returnnewobjectdeps(this.store,this.name,currentdeps.slice(),).proxy(currentstate)[key];}visited=true;this.deps.push(key);returnthis.proxy((this.snapshot=currentstate[key]));},});}}}
朴实无华,没有基于es6的proxy,因为兼容性不好。既然是前端的应用,自然是要照顾到es5的环境的,因此选择defineproerty是个不错的方案。
有了监听驱动,那监听岂不是易如反掌?
//假设user里的对象为:{firstname:'lady',lastname:'gaga'}constuserstate=store.getstate()['user'];functioncomputedfullname(){constproxy=newobjectdeps(store,'user').proxy(userstate);returnproxy.firstname '-' proxy.lastname;}constfullname=computedfullname();
现在我们看看collector里收集到多少个依赖
console.log(collector);//[objectdeps,objectdeps]
不错,两条依赖,第一条的deps链为['user', 'firstname'],第二条为['user', 'lastname']。
原理分析:
每次创建proxy时,构造函数均会执行collector.push(this)向采集器加入自己。
proxy访问firstname时,其实访问的是getter,getter中有一条this.deps.push(key)立即收集依赖,并返回下一级的proxy值。以此类推,即使是proxy.a.b.c.d这种深度操作也来者不拒,因为每次访问下一级都能收集依赖并合并到deps数组中。
proxy访问lastname时,由于proxy实例其实已经被firstname占用了(通过visited变量判断),所以getter逻辑中会直接返回一个新的objectdeps实例,此时lastname已经和我们看到的proxy变量没有任何关系了。
自动收集依赖已经实现了,我们试一下如何缓存属性
classobjectdeps{protectedsnapshot:any;proxy(){...}isdirty(){returnthis.snapshot!==this.getsnapshot();}protectedgetsnapshot(){constdeps=this.deps;letsnapshot=this.store.getstate();for(leti=0;i通过isdirty()的判断,即再次获得deps下的最新值和旧值做对比,便可以知道这个依赖是否为脏值。这一步便是缓存的关键。
现在你相信reselect是骗子了吧,明明可以自动依赖,非要多写几行代码增加心智负担?拜托,不是每个人都需要kpi压力的。
老师,我想直接在项目中使用上这个什么computed属性,应该去哪里找现成的呢?废话,当然是去山东找蓝翔。看看蓝翔大法:
import{definemodel,usecomputed}from'foca';exportconstusermodel=definemodel('user',{initialstate:{firstname:'lady',lastname:'gaga',},computed:{//清爽fullname(){returnthis.state.firstname '-' this.state.lastname;},},});//app.tsxconstapp:fc=()=>{constfullname=usecomputed(usermodel.fullname);return{fullname};};到此,关于“怎么用redux实现computed计算属性”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注恰卡编程网网站,小编会继续努力为大家带来更多实用的文章!
展开全文