koa中间件的实现原理如何

这篇文章将为大家详细讲解有关koa中间件的实现原理如何,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。

10年积累的网站设计、成都做网站经验,可以快速应对客户对网站的新想法和需求。提供各种问题对应的解决方案。让选择我们的客户得到更好、更有力的网络服务。我虽然不认识你,你也不认识我。但先做网站设计后付款的网站建设流程,更有尚志免费网站建设让你可以放心的选择与我们合作。

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

问题是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是用户传入的。

这里面有一些亮点:

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

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

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

关于“koa中间件的实现原理如何”这篇文章就分享到这里了,希望以上内容可以对大家有一定的帮助,使各位可以学到更多知识,如果觉得文章不错,请把它分享出去让更多的人看到。


文章标题:koa中间件的实现原理如何
分享链接:http://myzitong.com/article/joogdi.html