本文是一块儿学习造轮子系列的第二篇,本篇咱们将从零开始写一个小巧完整的Redux,本系列文章将会选取一些前端比较经典的轮子进行源码分析,而且从零开始逐步实现,本系列将会学习Promises/A+,Redux,react-redux,vue,dom-diff,webpack,babel,kao,express,async/await,jquery,Lodash,requirejs,lib-flexible等前端经典轮子的实现方式,每一章源码都托管在github上,欢迎关注~
相关系列文章:
一块儿学习造轮子(一):从零开始写一个符合Promises/A+规范的promise
一块儿学习造轮子(二):从零开始写一个Redux
一块儿学习造轮子(三):从零开始写一个React-Redux
本系列github仓库:
一块儿学习造轮子系列github(欢迎star~)html
Redux是JavaScript状态容器,提供可预测化的状态管理。本文将会详细介绍Redux五个核心方法
createStore,applyMiddleware,bindActionCreators,combineReducers,compose的实现原理,最后将本身封装一个小巧完整的redux库,随后会介绍一下常常与Redux一块儿结合使用的Redux经常使用中间件redux-logger,redux-thunk,redux-promise等中间件的实现原理。
前端
本文对于Redux是什么及Redux几个核心方法如何使用只会作简单介绍,若是还没用过Redux建议先学习基础知识。
vue
推荐文章:
Redux 入门教程(一):基本用法
Redux 入门教程(二):中间件与异步操做
Redux 入门教程(三):React-Redux 的用法
react
本文全部代码在github建有代码仓库,能够点此查看本文代码,也欢迎你们star~jquery
首先,咱们先来看一种使用Redux的基础场景:webpack
function reducer(state, action) {} const store = createStore(reducer) //用reducer生成了store store.subscribe(() => renderApp(store.getState())) //注册state变化的回调 renderApp(store.getState()) //初始化页面 store.dispatch(xxxaction) //发出action
上面代码是一个用到Redux的基础场景,首先定义了一个reducer,而后用这个reducer生成了store,在store上注册当state发生变化后要执行的回调函数,而后使用初始state先渲染一下页面,当页面有操做时,store.dispatch发出一个action,action和旧的state通过reducer计算生成新的state,此时state变化,触发回调函数使用新的state从新渲染页面,这个简单的场景囊括了整个redux工做流,
如图所示:
这个场景主要用到Redux里面的createStore方法,这是Redux里最核心的方法,下面咱们简单实现一下这个方法。git
function createStore(reducer) { let state = null //用来存储全局状态 let listeners = [] //用来存储状态发生变化的回调函数数组 const subscribe = (listener) => { //用来注册回调函数 listeners.push(listener) } const getState = () => state //用来获取最新的全局状态 const dispatch = (action) => { //用来接收一个action,并利用reducer,根据旧的state和action计算出最新的state,而后遍历回调函数数组,执行回调. state = reducer(state, action) //生成新state listeners.forEach((listener) => listener()) //执行回调 } dispatch({}) //初始化全局状态 return { getState, dispatch, subscribe } //返回store对象,对象上有三个方法供外部使用 }
其实实现这个方法并不复杂github
通过以上三步,咱们便实现了一个简单的createStore方法。web
咱们在开发稍微大一些的项目时reducer通常有多个,咱们会通常会创建一个reducers文件夹,里面存储项目中用到的全部reducer,而后使用一个combineReducers方法将全部reducer合并成一个传给createStore方法。express
import userInfoReducer from './userinfo.js' import bannerDataReducer from './banner.js' import recordReducer from './record.js' import clientInfoReducer from './clicentInfo.js' const rootReducer = combineReducers({ userInfoReducer, bannerDataReducer, recordReducer, clientInfoReducer }) const store = createStore(rootReducer)
接下来,咱们就一块儿来实现combineReducers这个方法:
const combineReducers = reducers => (state = {}, action) => { let currentState = {}; for (let key in reducers) { currentState[key] = reducers[key](state[key], action); } return currentState; };
{userInfoReducer,bannerDataReducer}
,userInfoReducer里state原本是这样:{userId:1,name:"张三"}
,而bannerDataReducer里的state原本是{pictureId:1,pictureUrl:"http://abc.com/1.jpg"}
{ userInfoReducer: { userId: 1, name: "张三" }, bannerDataReducer: { pictureId: 1, pictureUrl: "http://abc.com/1.jpg" } }
到此咱们实现了第二个方法combineReducers。
接下来介绍bindActionCreators这个方法,这是redux提供的一个辅助方法,可以让咱们以方法的形式来调用action。同时,自动dispatch对应的action。它接收2个参数,第一个参数是接收一个action creator,第二个参数接收一个 dispatch 函数,由 Store 实例提供。
好比说咱们有一个TodoActionCreators
export function addTodo(text) { return { type: 'ADD_TODO', text }; } export function removeTodo(id) { return { type: 'REMOVE_TODO', id }; }
咱们以前须要这样使用:
import * as TodoActionCreators from './TodoActionCreators'; let addReadAction = TodoActionCreators.addTodo('看书'); dispatch(addReadAction); let addEatAction = TodoActionCreators.addTodo('吃饭'); dispatch(addEatAction); let removeEatAction = TodoActionCreators.removeTodo('看书'); dispatch(removeEatAction);
如今只须要这样:
import * as TodoActionCreators from './TodoActionCreators'; let TodoAction = bindActionCreators(TodoActionCreators, dispatch); TodoAction.addTodo('看书') TodoAction.addTodo('吃饭') TodoAction.removeTodo('看书')
好了,说完了如何使用,咱们来实现一下这个方法
function bindActionCreator(actions, dispatch) { let newActions = {}; for (let key in actions) { newActions[key] = () => dispatch(actions[key].apply(null, arguments)); } return newActions; }
方法实现也不难,就是遍历ActionCreators里面的全部action,每一个都使用一个函数进行包裹dispatch行为并将这些函数挂载到一个对象上对外暴露,当咱们在外部的调用这个函数的时候,就会自动的dispatch对应的action,这个方法的实现其实也是利用了闭包的特性。
这个方法在使用react-redux里面常常见到,等讲react-redux实现原理时会再说一下。
最后,还剩两个方法,一个是compose,一个是applyMiddleware,这两个都是使用redux中间件时要用到的方法,先来讲说compose这个方法,这是一个redux里的辅助方法,其做用是把一系列的函数,组装生成一个新的函数,而且从后到前依次执行,后面函数的执行结果做为前一个函数执行的参数。
好比说咱们有这样几个函数:
function add1(str) { return str + 1 } function add2(str) { return str + 2 } function add3(str) { return str + 3 }
咱们想依次执行函数,并把执行结果传到下一层就要像下面同样一层套一层的去写:
let newstr = add3(add2(add1("abc"))) //"abc123"
这只是3个,若是数量多了或者数量不固定处理起来就很麻烦,可是咱们用compose写起来就很优雅:
let newaddfun = compose(add3, add2, add1); let newstr = newaddfun("abc") //"abc123"
那compose内部是如何实现的呢?
function compose(...funcs) { return funcs.reduce((a, b) => (...args) => a(b(...args))); }
其实核心代码就一句,这句代码使用了reduce方法巧妙地将一系列函数转为了add3(add2(add1(...args)))
这种形式,咱们使用上面的例子一步一步地拆分看一下,当调用compose(add3, add2, add1)
,funcs是add3, add2, add1,第一次进入时a是add3,b是add2,展开就是这样子:(add3, add2)=>(...args)=>add3(add2(...args))
,传入了add3, add2,返回一个这样的函数(...args)=>add3(add2(...args))
,而后reduce继续进行,第二次进入时a是上一步返回的函数(...args)=>add3(add2(...args))
,b是add1,因而执行到a(b(...args)))
时,b(...args)
做为a函数的参数传入,变成了这种形式:(...args)=>add3(add2(add1(...args)))
,是否是很巧妙。
最后咱们来看最后一个方法applyMiddleware,咱们在redux项目中,使用中间件时通常这样写:
import thunk from 'redux-thunk' import logger from 'redux-logger' const middleware = [thunk, logger] const store = createStore(rootReducer, applyMiddleware(...middleware))
上面咱们用到了thunk和logger这两个中间件,在createStore建立仓库时传入一个新的参数applyMiddleware(...middleware),在此告诉redux咱们要使用的中间件,因此咱们要先改造一下createStore方法,让其支持中间件参数的传入。
function createStore(reducer, enhancer) { //若是传入了中间件函数,使用中间件加强createStore方法 if (typeof enhancer === 'function') { return enhancer(createStore)(reducer) } let state = null const listeners = [] const subscribe = (listener) => { listeners.push(listener) } const getState = () => state const dispatch = (action) => { state = reducer(state, action) listeners.forEach((listener) => listener()) } dispatch({}) return { getState, dispatch, subscribe } }
而后接下来以redux-logger中间件为例来分析一下redux中间件的实现方式。
首先咱们能够先思考一下,若是咱们不用logger中间件,想实现logger的功能该怎样作呢?
let store = createStore(reducer); let dispatch = store.dispatch; store.dispatch = function (action) { console.log(store.getState()); dispatch(action); console.log(store.getState()) };
咱们能够在原始dispatch方法外面包装一层函数,让发起真正的dispatch以前和以后都打印一下日志,调用时调用包装后的这个dispatch函数,其实redux中间件原理的思路就是这样的:将store的dispatch进行替换,换成一个功能加强了可是仍然具备dispach功能的新函数。
那applyMiddleware方法里是如何改造dispatch来加强功能的呢?首先咱们来看个简单版本,假如咱们只有一个中间件,如何实现applyMiddleware方法呢?
function applyMiddleware(middleware) { return function a1(createStore) { return function a2(reducer) { //取出原始dispatch方法 const store = createStore(reducer) let dispatch = store.dispatch //包装dispatch const middlewareAPI = { getState: store.getState, dispatch: (action) => dispatch(action) } let mid = middleware(middlewareAPI) dispatch = mid(store.dispatch) //使用包装后的dispatch覆盖store.dispatch返回新的store对象 return { ...store, dispatch } } } } //中间件 let logger = function({ dispatch, getState }) { return function l1(next) { return function l2(action) { console.log(getState()); next(action) console.log(getState()) } } } //reducer函数 function reducer(state, action) { if (!state) state = { count: 0 } console.log(action) switch (action.type) { case 'add': let obj = {...state, count: ++state.count } return obj; case 'sub': return {...state, count: --state.count } default: return state } } const store = createStore(reducer, applyMiddleware(logger))
首先咱们定义了的applyMiddleware方法,它接收一个中间件做为参数。而后定义了一个logger中间件函数,它接收dispatch和getState方法以供内部使用。这两个函数Redux源码里都是使用高阶函数实现的,在这里与源码保持一致也使用高阶函数实现,可是为了方便理解,使用具名的function函数代替匿名箭头函数能够看得更清晰。
当咱们执行const store = createStore(reducer,applyMiddleware(logger))
时,首先applyMiddleware(logger)
执行,将logger存在闭包里,而后返回了一个接收createStore方法的函数a1,将a1这个函数做为第二个参数传入createStore方法,由于传入了第二个参数,因此createstore里面其实会执行这一段代码:
if (typeof enhancer === 'function') { return enhancer(createStore)(reducer) }
当执行return enhancer(createStore)(reducer)
,其实执行的是a1(createStore)(reducer)
,当执行a1(createStore)
时返回a2,最后return的是a2(reducer)
的执行结果。
而后,咱们看看a2内部都作了些什么,我给这个函数定义了三个阶段,首先为取出原始dispatch阶段,这一阶段执行createStore(reducer)
方法,并拿出原始的dispatch方法。
接着,咱们到了第二个阶段包装原始dispatch,首先咱们定义了middlewareAPI用来给中间件函数使用,这里的getState直接使用了store.getState,而dispatch使用函数包了一层,(action)=>dispatch(action)
,为何呢,由于咱们最终要给中间件使用的dispatch方法,必定是通过各类中间件包装后的dispatch方法,而不是原方法,因此咱们这里将dispatch方法设置为一个变量。而后将middlewareAPI传入middleware执行,返回一个函数mid(也就是logger里面的l1),这个函数接收一个next方法做为参数,而后当咱们执行dispatch = mid(store.dispatch)
时,将store.dispatch做为next方法传入,并把返回的函数l2做为新的dispatch,咱们能够看到新的dispatch方法其实里面作了和咱们上面本身直接改造store.dispatch作了一样的事情:
function l2(action) { console.log(getState()); next(action) console.log(getState()) }
都是接收一个action,先打印日志,而后执行原始的dispatch方法去发一个action,而后再打印日志。
最后到了第三个阶段:使用包装后的dispatch覆盖store.dispatch方法后返回新的store对象。
到此,当咱们在外面执行store.dispatch({type:add})时,实际上执行的是包装后的dispatch方法,因此logger中间件就生效了,如图所示真正发起dispatch的先后都打印出了最新状态:
如今咱们在上一版applyMiddleware的基础上再改造,使其支持多个中间件:
import compose from './compose'; function applyMiddleware(...middlewares) { return function a1(createStore) { return function a2(reducer) { const store = createStore(reducer) let dispatch = store.dispatch let chain = [] const middlewareAPI = { getState: store.getState, dispatch: (action) => dispatch(action) } chain = middlewares.map(middleware => middleware(middlewareAPI)) dispatch = compose(...chain)(store.dispatch) return { ...store, dispatch } } } } let loggerone = function({ dispatch, getState }) { return function loggerOneOut(next) { return function loggerOneIn(action) { console.log("loggerone:", getState()); next(action) console.log("loggerone:", getState()) } } } let loggertwo = function({ dispatch, getState }) { return function loggerTwoOut(next) { return function loggerTwoIn(action) { console.log("loggertwo:", getState()); next(action) console.log("loggertwo:", getState()) } } } const store = createStore(reducer, applyMiddleware([loggertwo, loggerone]))
首先当调用applyMiddleware方法时,由传入一个中间件变为传入一个中间件数组。
而后咱们在applyMiddleware方法中维护一个chain数组,这个数组用于存储中间件链。
当执行到 chain = middlewares.map(middleware => middleware(middlewareAPI))
时,chain里面存放的是[loggerTwoOut,loggerOneOut]
。
而后下一步咱们改造dispatch时用到了咱们以前讲过的compose方法,dispatch=compose(...chain)(store.dispatch)
其实至关因而执行了dispatch =loggerTwoOut(loggerOneOut(store.dispatch))
,而后这一句loggerTwoOut(loggerOneOut(store.dispatch))
再次拆开看一下是如何执行的,当执行loggerOneOut(store.dispatch)
,返回loggerOneIn函数,并将store.dispatch方法做为loggerOneIn里面的next方法。如今函数变成了这样:loggerTwoOut(loggerOneIn)
,当执行这一句时,返回loggerTwoIn函数,并将loggerOneIn做为loggerTwoIn方法里的next方法。最后给dispatch赋值:dispatch =loggerTwoIn
。
在外部咱们调用store.dispatch({type:add})
时,实际执行的是loggerTwoIn({type:add})
,因此会先执行 console.log("loggertwo:", getState())
,而后执行next(action)
时执行的实际上是loggerOneIn(action)
,进入到loggerOneIn内部,因此会执行console.log("loggerone:",getState())
;而后执行next(action)
,这里的其实执行的是原始的store.dispatch方法,因此会真正的把action提交,提交完后继续执行,执行console.log("loggerone:",getState())
,而后loggerOneIn执行完毕,执行权交还到上一层loggerTwoIn,loggerTwoIn继续执行,执行console.log("loggertwo:", getState())
,结束。
画一张图形象的表示下执行流程:
到此,applymiddleware方法就讲完了,咱们来看下redux官方源码的实现:
function applyMiddleware(...middlewares) { return (createStore) => (reducer, preloadedState, enhancer) => { const store = createStore(reducer, preloadedState, enhancer) let dispatch = store.dispatch let chain = [] const middlewareAPI = { getState: store.getState, dispatch: (action) => dispatch(action) } chain = middlewares.map(middleware => middleware(middlewareAPI)) dispatch = compose(...chain)(store.dispatch) return { ...store, dispatch } } }
咱们实现的applyMiddleware方法对比官方除了没有对先后端同构时预取数据preloadedState作支持外,其他功能都完整实现了。
到此咱们把redux里全部方法都实现了一遍,固然咱们实现的只是每一个方法最核心最经常使用的部分,并无将redux源码逐字逐句去翻译。由于我的认为对于源码的学习应该抓住主线,学习源码中的核心代码及闪光点,若是对redux其余功能感兴趣的,能够自行看官方源码学习。
接下来,咱们将redux经常使用的三个中间件来实现一下
let logger = function({ dispatch, getState }) { return function(next) { return function(action) { console.log(getState()); next(action) console.log(getState()) } } }
这个咱们上面讲applyMiddleware时已经讲过了,再也不多说。
redux-thunk在咱们日常使用时主要用来处理异步提交action状况,引入了redux-thunk后咱们能够异步提交action
const fetchPosts = postTitle => (dispatch, getState) => { dispatch(requestPosts(postTitle)); return fetch(`/some/API/${postTitle}.json`) .then(response => response.json()) .then(json => dispatch(receivePosts(postTitle, json))); }; store.dispatch(fetchPosts('reactjs'))
咱们能够看到fetchPosts('reactjs')返回的是一个函数,而redux里的dispatch方法不能接受一个函数,Redux官方源码中明确说了,action必须是一个纯粹的对象,处理异步action时须要使用中间件,
function dispatch(action) { if (!isPlainObject(action)) { throw new Error( 'Actions must be plain objects. ' + 'Use custom middleware for async actions.' ) } ...... }
那redux-thunk到底作了什么使dispatch能够传入函数呢?
let thunk = function({ getState, dispatch }) { return function(next) { return function(action) { if (typeof action == 'function') { action(dispatch, getState); } else { next(action); } } } }
thunk中间件在内部进行判断,若是传入了一个函数,就去执行它,不是函数就无论交给下一个中间件,以上面的fetchPosts为例,当执行store.dispatch(fetchPosts('reactjs'))
时,给dispatch传入了一个函数:
postTitle => (dispatch, getState) => { dispatch(requestPosts(postTitle)); return fetch(`/some/API/${postTitle}.json`) .then(response => response.json()) .then(json => dispatch(receivePosts(postTitle, json))); };
thunk中间件发现是个函数,因而执行它,先发出一个Action(requestPosts(postTitle)),而后进行异步操做。拿到结果后,先将结果转成 JSON 格式,而后再发出一个Action(receivePosts(postTitle,json))。这两个Action都是普通对象,因此当dispatch时会走else {next(action);}这个分支,继续执行.这样就解决了dispatch不能接受函数的问题。
最后讲一个redux-promise中间件.dispatch目前能够支持传入函数了,利用redux-promise咱们再让它支持传入promise对象,平时咱们在用这个中间件时,通常有两种用法:
写法一,返回值是一个 Promise 对象。
const fetchPosts = (dispatch, postTitle) => new Promise(function(resolve, reject) { dispatch(requestPosts(postTitle)); return fetch(`/some/API/${postTitle}.json`) .then(response => { type: 'FETCH_POSTS', payload: response.json() }); });
写法二,Action 对象的payload属性是一个Promise对象。这须要从redux里引入createAction方法,而且写法也要变成下面这样。
import { createAction } from 'redux-actions'; class AsyncApp extends Component { componentDidMount() { const { dispatch, selectedPost } = this.props // 发出同步 Action dispatch(requestPosts(selectedPost)); // 发出异步 Action dispatch(createAction( 'FETCH_POSTS', fetch(`/some/API/${postTitle}.json`) .then(response => response.json()) )); } }
让咱们来实现一下redux-promise中间件:
let promise = function({ getState, dispatch }) { return function(next) { return function(action) { if (action.then) { action.then(dispatch); } else if (action.payload && action.payload.then) { action.payload.then(payload => dispatch({...action, payload }), payload => dispatch({...action, payload })); } else { next(action); } } } }
咱们实现redux-thunk时是判断若是传入function就执行这个function,不然next(action)继续执行;redux-promise同理,当action或action的payload上面有then方法时,咱们认为它是promise对象,就让dispatch到promise的then里面再执行,直到dispatch提交的action没有then方法,咱们认为它不是promise了,能够执行next(action)交给下一个中间件执行了。
本篇介绍了Redux五个方法createStore,applyMiddleware,bindActionCreators,combineReducers,compose的实现原理,并本身封装了一个小巧完整的Redux库,同时简单介绍了Redux里经常使用的3个中间件redux-logger,redux-thunk,redux-promise的实现原理,本文全部代码在github建有代码仓库,能够点击查看本文源码。
与Redux相关的比较经典的轮子还有React-Redux和redux-saga,因本文篇幅如今已经很长,因此这两个轮子的实现将放到后续的一块儿学习造轮子系列中,敬请关注~