24行JavaScript代码实现Redux的方法实例
前言
Redux是迄今为止创建的最重要的JavaScript库之一,灵感来源于以前的艺术比如Flux和Elm,Redux通过引入一个包含三个简单要点的可伸缩体系结构,使得JavaScript函数式编程成为可能。如果你是初次接触Redux,可以考虑先阅读官方文档。
1.Redux大多是规约
考虑如下这个使用了Redux架构的简单的计数器应用。如果你想跳过的话可以直接查看GithubRepo。
1.1State存储在一棵树中
该应用程序的状态看起来如下:
constinitialState={count:0};
1.2Action声明状态更改
根据Redux规约,我们不直接修改(突变)状态。
//在Redux应用中不要做如下操作 state.count=1;
相反,我们创建在应用中用户可能用到的所有行为。
constactions={ increment:{type:'INCREMENT'}, decrement:{type:'DECREMENT'} };
1.3Reducer解释行为并更新状态
在最后一个架构部分我们叫做Reduer,其作为一个纯函数,它基于以前的状态和行为返回状态的新副本。
- 如果increment被触发,则增加state.count
- 如果decrement被触发,则减少state.count
constcountReducer=(state=initialState,action)=>{ switch(action.type){ caseactions.increment.type: return{ count:state.count+1 }; caseactions.decrement.type: return{ count:state.count-1 }; default: returnstate; } };
1.4目前为止还没有Redux
你注意到了吗?到目前为止我们甚至还没有接触到Redux库,我们仅仅只是创建了一些对象和函数,这就是为什么我称其为"大多是规约",90%的Redux应用其实并不需要Redux。
2.开始实现Redux
要使用这种架构,我们必须要将它放入到一个store当中,我们将仅仅实现一个函数:createStore。使用方式如下:
import{createStore}from'redux' conststore=createStore(countReducer); store.subscribe(()=>{ console.log(store.getState()); }); store.dispatch(actions.increment); //logs{count:1} store.dispatch(actions.increment); //logs{count:2} store.dispatch(actions.decrement); //logs{count:1}
下面这是我们的初始化样板代码,我们需要一个监听器列表listeners和reducer提供的初始化状态。
constcreateStore=(yourReducer)=>{ letlisteners=[]; letcurrentState=yourReducer(undefined,{}); }
无论何时某人订阅了我们的store,那么他将会被添加到listeners数组中。这是非常重要的,因为每次当某人在派发(dispatch)一个动作(action)的时候,所有的listeners都需要在此次事件循环中被通知到。调用yourReducer函数并传入一个undefined和一个空对象将会返回一个initialState,这个值也就是我们在调用store.getState()时的返回值。既然说到这里了,我们就来创建这个方法。
2.1store.getState()
这个函数用于从store中返回最新的状态,当用户每次点击一个按钮的时候我们都需要最新的状态来更新我们的视图。
constcreateStore=(yourReducer)=>{ letlisteners=[]; letcurrentState=yourReducer(undefined,{}); return{ getState:()=>currentState }; }
2.2store.dispatch()
这个函数使用一个action作为其入参,并且将这个action和currentState反馈给yourReducer来获取一个新的状态,并且dispatch方法还会通知到每一个订阅了当前store的监听者。
constcreateStore=(yourReducer)=>{ letlisteners=[]; letcurrentState=yourReducer(undefined,{}); return{ getState:()=>currentState, dispatch:(action)=>{ currentState=yourReducer(currentState,action); listeners.forEach((listener)=>{ listener(); }); } }; };
2.3store.subscribe(listener)
这个方法使得你在当store接收到一个action的时候能够被通知到,可以在这里调用store.getState()来获取最新的状态并更新UI。
constcreateStore=(yourReducer)=>{ letlisteners=[]; letcurrentState=yourReducer(undefined,{}); return{ getState:()=>currentState, dispatch:(action)=>{ currentState=yourReducer(currentState,action); listeners.forEach((listener)=>{ listener(); }); }, subscribe:(newListener)=>{ listeners.push(newListener); constunsubscribe=()=>{ listeners=listeners.filter((l)=>l!==newListener); }; returnunsubscribe; } }; };
同时subscribe函数返回了另一个函数unsubscribe,这个函数允许你当不再对store的更新感兴趣的时候能够取消订阅。
3.整理代码
现在我们添加按钮的逻辑,来看看最后的源代码:
//简化版createStore函数 constcreateStore=(yourReducer)=>{ letlisteners=[]; letcurrentState=yourReducer(undefined,{}); return{ getState:()=>currentState, dispatch:(action)=>{ currentState=yourReducer(currentState,action); listeners.forEach((listener)=>{ listener(); }); }, subscribe:(newListener)=>{ listeners.push(newListener); constunsubscribe=()=>{ listeners=listeners.filter((l)=>l!==newListener); }; returnunsubscribe; } }; }; //Redux的架构组成部分 constinitialState={count:0}; constactions={ increment:{type:'INCREMENT'}, decrement:{type:'DECREMENT'} }; constcountReducer=(state=initialState,action)=>{ switch(action.type){ caseactions.increment.type: return{ count:state.count+1 }; caseactions.decrement.type: return{ count:state.count-1 }; default: returnstate; } }; conststore=createStore(countReducer); //DOM元素 constincrementButton=document.querySelector('.increment'); constdecrementButton=document.querySelector('.decrement'); //给按钮添加点击事件 incrementButton.addEventListener('click',()=>{ store.dispatch(actions.increment); }); decrementButton.addEventListener('click',()=>{ store.dispatch(actions.decrement); }); //初始化UI视图 constcounterDisplay=document.querySelector('h1'); counterDisplay.innerHTML=parseInt(initialState.count); //派发动作的时候跟新UI store.subscribe(()=>{ conststate=store.getState(); counterDisplay.innerHTML=parseInt(state.count); });
我们再次看看最后的视图效果:
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对毛票票的支持。
原文:https://www.freecodecamp.org/news/redux-in-24-lines-of-code/
作者:YazeedBzadough
译者:小维FE
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。