千锋教育-做有情怀、有良心、有品质的职业教育机构

手机站
千锋教育

千锋学习站 | 随时随地免费学

千锋教育

扫一扫进入千锋手机站

领取全套视频
千锋教育

关注千锋学习站小程序
随时随地免费学习课程

当前位置:首页  >  技术干货  > 浅谈koa中间件的实现原理

浅谈koa中间件的实现原理

来源:千锋教育
发布人:qyf
时间: 2022-10-10 15:29:38 1665386978

  koa中间件的实现原理如何?先来看一个例子。

  koa的执行顺序是这样的:

  const middleware = async function (ctx, next) {

  console.log(1)

  await next()

  console.log(6)

  }

  const middleware2 = async function (ctx, next) {

  console.log(2)

  await next()

  console.log(5)

  }

  const middleware3 = async function (ctx, next) {

  console.log(3)

  await next()

  console.log(4)

  }

  会依次打印1,2,3,4,5,6

图片12

  问题是koa中间件实现原理,也就是洋葱模型的实现原理是什么?

  一、问题分析

  async await是promise的语法糖,await后面跟一个promise,所以上面的代码可以写成:

  const middleware = function (ctx, next) {

  console.log(1)

  next().then(() => {

  console.log(6)

  })

  }

  const middleware2 = function (ctx, next) {

  console.log(2)

  next().then(() => {

  console.log(5)

  })

  }

  const middleware3 = function (ctx, next) {

  console.log(3)

  next().then(() => {

  console.log(4)

  })

  }

  改成这样更好理解一些,所以流程控制的核心在于next的实现。

  next要求调用队列中下一个middleware,当达到最后一个的时候resolve。这样最后面的promise先resolve,一直到第一个,这样就是洋葱模型的顺序了。

  二、实现

  koa-compose的实现是这样的:

  function compose(middleware) {

  return function (context, next) {

  let index = -1

  return dispatch(0)

  function dispatch(i) {

  index = i

  let fn = middleware[i]

  if (i === middleware.length) fn = next

  if (!fn) return Promise.resolve()

  try {

  return Promise.resolve(fn(context, dispatch.bind(null, i + 1)))

  } catch (err) {

  return Promise.reject(err)

  }

  }

  }

  }

  我们把一些参数检查的非核心逻辑去掉了,实现代码就上面那些。每次传入的next都是调用下一个middleware,这样是一个递归的过程,结束条件是最后一个middleware的next是用户传入的。

  这里面有一些亮点:

  1. 这是一种尾递归的形式,尾递归的特点是最后返回的值是一个递归的函数调用,这样执行完就会在调用栈中销毁,不会占据调用栈.

  2. 返回的是一个Promise.resolve包装之后的调用,而不是同步的调用,所以这是一个异步递归,异步递归比同步递归的好处是可以被打断,如果中间有一些优先级更高的微任务,那么可以先执行别的微任务

  3. compose是函数复合,把n个middleware复合成一个,参数依然是context和next,这种复合之后依然是一个middleware,还可以继续进行复合。

  三、总结

  Koa 中间件的实现原理,也就是洋葱模型的实现原理,核心在于next的实现。next需要依次调用下一个middleware,当到最后一个的时候结束,这样后面middleware的promise先resolve,然后直到第一个,这样的流程也就是洋葱模型的流程了。

  实现的时候还有一些细节,一个是递归最好做成尾递归的形式,而是用异步递归而不是同步递归,第三就是形式上用函数复合的形式,这样复合之后的中间件还可以继续复合。

tags:
声明:本站稿件版权均属千锋教育所有,未经许可不得擅自转载。
10年以上业内强师集结,手把手带你蜕变精英
请您保持通讯畅通,专属学习老师24小时内将与您1V1沟通
免费领取
今日已有369人领取成功
刘同学 138****2860 刚刚成功领取
王同学 131****2015 刚刚成功领取
张同学 133****4652 刚刚成功领取
李同学 135****8607 刚刚成功领取
杨同学 132****5667 刚刚成功领取
岳同学 134****6652 刚刚成功领取
梁同学 157****2950 刚刚成功领取
刘同学 189****1015 刚刚成功领取
张同学 155****4678 刚刚成功领取
邹同学 139****2907 刚刚成功领取
董同学 138****2867 刚刚成功领取
周同学 136****3602 刚刚成功领取
相关推荐HOT