const Koa = require('koa'); const app = new Koa(); app.use(async (ctx, next) => { fn1(); await next(); fn2(); }); app.use(async (ctx, next) => { fn3(); await next(); fn4() }); app.use(async (ctx, next) => { fn5(); await next(); fn6() }); app.listen(3000);
執行順序 fn1 -> fn3 -> fn5 -> fn6 -> fn4 -> fn2
每當執行next時,執行下一個中間件,執行到最後一箇中間件後開始往回執行javascript
koa源碼最重要的部分,如何實現中間件級聯,如下是compose方法的源碼(爲方便觀看,部分進行了ES6處理)java
function compose (middleware) { if (!Array.isArray(middleware)) throw new TypeError('Middleware stack must be an array!') for (const fn of middleware) { if (typeof fn !== 'function') throw new TypeError('Middleware must be composed of functions!') } return function (context, next) { // last called middleware # let index = -1 return dispatch(0) function dispatch (i) { if (i <= index) return Promise.reject(new Error('next() called multiple times')) index = i let fn = middleware[i] if (i === middleware.length) fn = next // next = undefined if (!fn) return Promise.resolve() // 當運行到最後一個middleware時結束 try { return Promise.resolve( fn(context, next = () => dispatch(i + 1)) ) } catch (err) { return Promise.reject(err) } } } }