返回顶部
首页 > 资讯 > 精选 >怎么更好地理解中间件和洋葱模型
  • 861
分享到

怎么更好地理解中间件和洋葱模型

2023-06-16 16:06:15 861人浏览 薄情痞子
摘要

这篇文章主要介绍“怎么更好地理解中间件和洋葱模型”,在日常操作中,相信很多人在怎么更好地理解中间件和洋葱模型问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么更好地理解中间件和洋葱模型”的疑惑有所帮助!接下来

这篇文章主要介绍“怎么更好地理解中间件和洋葱模型”,在日常操作中,相信很多人在怎么更好地理解中间件和洋葱模型问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么更好地理解中间件和洋葱模型”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

一、Koa 中间件

在 @types/koa-compose 包下的 index.d.ts 头文件中我们找到了中间件类型的定义:

// @types/koa-compose/index.d.ts declare namespace compose {   type Middleware<T> = (context: T, next: Koa.Next) => any;   type ComposedMiddleware<T> = (context: T, next?: Koa.Next) => Promise<void>; }    // @types/koa/index.d.ts => Koa.Next type Next = () => Promise<any>;

通过观察 Middleware 类型的定义,我们可以知道在 Koa 中,中间件就是普通的函数,该函数接收两个参数:context 和 next。其中  context 表示上下文对象,而 next 表示一个调用后返回 Promise 对象的函数对象。

了解完 Koa 的中间件是什么之后,我们来介绍 Koa 中间件的核心,即 compose 函数:

function wait(ms) {   return new Promise((resolve) => setTimeout(resolve, ms || 1)); }  const arr = []; const stack = [];  // type Middleware<T> = (context: T, next: Koa.Next) => any; stack.push(async (context, next) => {   arr.push(1);   await wait(1);   await next();   await wait(1);   arr.push(6); });  stack.push(async (context, next) => {   arr.push(2);   await wait(1);   await next();   await wait(1);   arr.push(5); });  stack.push(async (context, next) => {   arr.push(3);   await wait(1);   await next();   await wait(1);   arr.push(4); });  await compose(stack)({});

对于以上的代码,我们希望执行完 compose(stack)({}) 语句之后,数组 arr 的值为 [1, 2, 3, 4, 5,  6]。这里我们先不关心 compose 函数是如何实现的。我们来分析一下,如果要求数组 arr 输出期望的结果,上述 3 个中间件的执行流程:

开始执行第 1 个中间件,往 arr 数组压入 1,此时 arr 数组的值为 [1],接下去等待 1 毫秒。为了保证 arr 数组的第 1 项为  2,我们需要在调用 next 函数之后,开始执行第 2 个中间件。

开始执行第 2 个中间件,往 arr 数组压入 2,此时 arr 数组的值为 [1, 2],继续等待 1 毫秒。为了保证 arr 数组的第 2 项为  3,我们也需要在调用 next 函数之后,开始执行第 3 个中间件。

开始执行第 3 个中间件,往 arr 数组压入 3,此时 arr 数组的值为 [1, 2, 3],继续等待 1 毫秒。为了保证 arr 数组的第 3  项为 4,我们要求在调用第 3 个中间的 next 函数之后,要能够继续往下执行。

当第 3 个中间件执行完成后,此时 arr 数组的值为 [1, 2, 3, 4]。因此为了保证 arr 数组的第 4 项为 5,我们就需要在第 3  个中间件执行完成后,返回第 2 个中间件 next 函数之后语句开始执行。

当第 2 个中间件执行完成后,此时 arr 数组的值为 [1, 2, 3, 4, 5]。同样,为了保证 arr 数组的第 5 项为 6,我们就需要在第  2 个中间件执行完成后,返回第 1 个中间件 next函数之后语句开始执行。

当第 1 个中间件执行完成后,此时 arr 数组的值为 [1, 2, 3, 4, 5, 6]。

为了更直观地理解上述的执行流程,我们可以把每个中间件当做 1 个大任务,然后在以 next 函数为分界点,在把每个大任务拆解为 3 个  beforeNext、next 和 afterNext 3 个小任务。

怎么更好地理解中间件和洋葱模型

在上图中,我们从中间件一的 beforeNext 任务开始执行,然后按照紫色箭头的执行步骤完成中间件的任务调度。在 77.9K 的 AxiOS  项目有哪些值得借鉴的地方 这篇文章中,阿宝哥从 任务注册、任务编排和任务调度 3 个方面去分析 Axios 拦截器的实现。同样,阿宝哥将从上述 3 个方面来分析  Koa 中间件机制。

1.1 任务注册

在 Koa 中,我们创建 Koa 应用程序对象之后,就可以通过调用该对象的 use 方法来注册中间件:

const Koa = require('koa'); const app = new Koa();  app.use(async (ctx, next) => {   const start = Date.now();   await next();   const ms = Date.now() - start;   console.log(`${ctx.method} ${ctx.url} - ${ms}ms`); });

其实 use 方法的实现很简单,在 lib/application.js 文件中,我们找到了它的定义:

// lib/application.js module.exports = class Application extends Emitter {     constructor(options) {     super();     // 省略部分代码      this.middleware = [];   }     use(fn) {    if (typeof fn !== 'function') throw new TypeError('middleware must be a function!');    // 省略部分代码     this.middleware.push(fn);    return this;   } }

由以上代码可知,在 use 方法内部会对 fn 参数进行类型校验,当校验通过时,会把 fn 指向的中间件保存到 middleware 数组中,同时还会返回  this 对象,从而支持链式调用。

1.2 任务编排

在 77.9K 的 Axios 项目有哪些值得借鉴的地方 这篇文章中,阿宝哥参考 Axios 拦截器的设计模型,抽出以下通用的任务处理模型:

怎么更好地理解中间件和洋葱模型

在该通用模型中,阿宝哥是通过把前置处理器和后置处理器分别放到 CoreWork 核心任务的前后来完成任务编排。而对于 Koa  的中间件机制来说,它是通过把前置处理器和后置处理器分别放到 await next() 语句的前后来完成任务编排


// 统计请求处理时长的中间件 app.use(async (ctx, next) => {   const start = Date.now();   await next();   const ms = Date.now() - start;   console.log(`${ctx.method} ${ctx.url} - ${ms}ms`); });

怎么更好地理解中间件和洋葱模型

1.3 任务调度

通过前面的分析,我们已经知道了,使用 app.use 方法注册的中间件会被保存到内部的 middleware 数组中。要完成任务调度,我们就需要不断地从  middleware 数组中取出中间件来执行。中间件的调度算法被封装到 koa-compose 包下的 compose 函数中,该函数的具体实现如下:

 function compose(middleware) {   // 省略部分代码   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;       if (!fn) return Promise.resolve();       try {         return Promise.resolve(fn(context, dispatch.bind(null, i + 1)));       } catch (err) {         return Promise.reject(err);       }     }   }; }

compose 函数接收一个参数,该参数的类型是数组,调用该函数之后会返回一个新的函数。接下来我们将以前面的例子为例,来分析一下 await  compose(stack)({}); 语句的执行过程。

1.3.1 dispatch(0)

怎么更好地理解中间件和洋葱模型

由上图可知,当在第一个中间件内部调用 next 函数,其实就是继续调用 dispatch 函数,此时参数 i 的值为 1。

1.3.2 dispatch(1)

怎么更好地理解中间件和洋葱模型

由上图可知,当在第二个中间件内部调用 next 函数,仍然是调用 dispatch 函数,此时参数 i 的值为 2。

1.3.3 dispatch(2)

怎么更好地理解中间件和洋葱模型

由上图可知,当在第三个中间件内部调用 next 函数,仍然是调用 dispatch 函数,此时参数 i 的值为 3。

1.3.4 dispatch(3)

怎么更好地理解中间件和洋葱模型

由上图可知,当 middleware 数组中的中间件都开始执行之后,如果调度时未显式地设置 next 参数的值,则会开始返回 next  函数之后的语句继续往下执行。当第三个中间件执行完成后,就会返回第二中间件 next 函数之后的语句继续往下执行,直到所有中间件中定义的语句都执行完成。

分析完 compose 函数的实现代码,我们来看一下 Koa 内部如何利用 compose 函数来处理已注册的中间件。

const Koa = require('koa'); const app = new Koa();  // 响应 app.use(ctx => {   ctx.body = '大家好,我是阿宝哥'; });  app.listen(3000);

利用以上的代码,我就可以快速启动一个服务器。其中 use 方法我们前面已经分析过了,所以接下来我们来分析 listen  方法,该方法的实现如下所示:

// lib/application.js module.exports = class Application extends Emitter {     listen(...args) {     debug('listen');     const server = Http.createServer(this.callback());     return server.listen(...args);   } }

很明显在 listen 方法内部,会先通过调用 node.js 内置 HTTP 模块的 createServer  方法来创建服务器,然后开始监听指定的端口,即开始等待客户端的连接。

另外,在调用 http.createServer 方法创建 HTTP 服务器时,我们传入的参数是  this.callback(),该方法的具体实现如下所示:

// lib/application.js const compose = require('koa-compose');  module.exports = class Application extends Emitter {     callback() {     const fn = compose(this.middleware);     if (!this.listenerCount('error')) this.on('error', this.onerror);      const handleRequest = (req, res) => {       const ctx = this.createContext(req, res);       return this.handleRequest(ctx, fn);     };     return handleRequest;   }

在 callback 方法内部,我们终于见到了久违的 compose 方法。当调用 callback 方法之后,会返回 handleRequest  函数对象用来处理 HTTP 请求。每当 Koa 服务器接收到一个客户端请求时,都会调用 handleRequest 方法,在该方法会先创建新的 Context  对象,然后在执行已注册的中间件来处理已接收的 HTTP 请求:

module.exports = class Application extends Emitter {     handleRequest(ctx, fnMiddleware) {     const res = ctx.res;     res.statusCode = 404;     const onerror = err => ctx.onerror(err);     const handleResponse = () => respond(ctx);     onFinished(res, onerror);     return fnMiddleware(ctx).then(handleResponse).catch(onerror);   } }

好的,Koa 中间件的内容已经基本介绍完了,对 Koa 内核感兴趣的小伙伴,可以自行研究一下。接下来我们来介绍洋葱模型及其应用。

二、洋葱模型2.1 洋葱模型简介

怎么更好地理解中间件和洋葱模型

(图片来源:https://eggjs.org/en/intro/egg-and-koa.html)

在上图中,洋葱内的每一层都表示一个独立的中间件,用于实现不同的功能,比如异常处理、缓存处理等。每次请求都会从左侧开始一层层地经过每层的中间件,当进入到最里层的中间件之后,就会从最里层的中间件开始逐层返回。因此对于每层的中间件来说,在一个  请求和响应 周期中,都有两个时机点来添加不同的处理逻辑。

2.2 洋葱模型应用

除了在 Koa 中应用了洋葱模型之外,该模型还被广泛地应用在 GitHub 上一些不错的项目中,比如 koa-router 和阿里巴巴的  midway、umi-request 等项目中。

介绍完 Koa 的中间件和洋葱模型,阿宝哥根据自己的理解,抽出以下通用的任务处理模型:

怎么更好地理解中间件和洋葱模型

上图中所述的中间件,一般是与业务无关的通用功能代码,比如用于设置响应时间的中间件:

// x-response-time async function responseTime(ctx, next) {   const start = new Date();   await next();   const ms = new Date() - start;   ctx.set("X-Response-Time", ms + "ms"); }

其实,对于每个中间件来说,前置处理器和后置处理器都是可选的。比如以下中间件用于设置统一的响应内容:

// response async function respond(ctx, next) {   await next();   if ("/" != ctx.url) return;   ctx.body = "Hello World"; }

尽管以上介绍的两个中间件都比较简单,但你也可以根据自己的需求来实现复杂的逻辑。Koa  的内核很轻量,麻雀虽小五脏俱全。它通过提供了优雅的中间件机制,让开发者可以灵活地扩展 WEB 服务器的功能,这种设计思想值得我们学习与借鉴。

到此,关于“怎么更好地理解中间件和洋葱模型”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!

--结束END--

本文标题: 怎么更好地理解中间件和洋葱模型

本文链接: https://lsjlt.com/news/284259.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • 怎么更好地理解中间件和洋葱模型
    这篇文章主要介绍“怎么更好地理解中间件和洋葱模型”,在日常操作中,相信很多人在怎么更好地理解中间件和洋葱模型问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”怎么更好地理解中间件和洋葱模型”的疑惑有所帮助!接下来...
    99+
    2023-06-16
  • 如何在PHP IDE中更好地处理文件数据类型?
    PHP是一种广泛应用于Web开发的编程语言,它支持各种数据类型,其中包括文件数据类型。在PHP中,文件数据类型是一个非常重要的数据类型,因为它可以被用来处理文件上传、文件读取、文件写入等操作。在本文中,我们将介绍如何在PHP的IDE中更好地...
    99+
    2023-09-20
    ide 文件 数据类型
  • Hibernate和模型对象怎么理解
    这篇文章主要讲解了“Hibernate和模型对象怎么理解”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Hibernate和模型对象怎么理解”吧!学习Hibernate时,经常会遇到一些小问题...
    99+
    2023-06-17
  • Python自然语言处理中的Path接口可以帮助我们更好地理解语言模型吗?
    Python是一种广泛使用的编程语言,其生态系统非常丰富,包括自然语言处理(NLP)领域。在NLP中,Path接口是一种非常有用的工具,可以帮助我们更好地理解语言模型。 Path接口是Python的一种文件路径操作库,它可以帮助我们更方便...
    99+
    2023-10-10
    自然语言处理 path 接口
  • Java中使用哪些数据类型可以更好地处理并发和日志?
    在Java开发中,处理并发和日志是非常重要的,因为这些功能对于许多应用程序而言是必需的。在Java中,有许多不同的数据类型可以用于处理并发和日志。在本文中,我们将探讨Java中使用哪些数据类型可以更好地处理并发和日志。 一、Java中的并发...
    99+
    2023-09-13
    数据类型 并发 日志
  • 机器学习中词袋模型和TF-IDF怎么理解
    本篇内容主要讲解“机器学习中词袋模型和TF-IDF怎么理解”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“机器学习中词袋模型和TF-IDF怎么理解”吧!示例我将用一个流行的例子来解释本文中的Bag...
    99+
    2023-06-19
  • koa中间件处理模块koa-compose怎么用
    这篇文章主要为大家展示了“koa中间件处理模块koa-compose怎么用”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“koa中间件处理模块koa-compos...
    99+
    2024-04-02
  • JavaScript中的原型和原型链怎么理解
    这篇文章主要介绍“JavaScript中的原型和原型链怎么理解”,在日常操作中,相信很多人在JavaScript中的原型和原型链怎么理解问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大...
    99+
    2024-04-02
  • Navicat中怎么导入和导出地理空间数据
    在 Navicat 中导入和导出地理空间数据,可以按照以下步骤进行操作: 导出地理空间数据: 在 Navicat 中连接到数据...
    99+
    2024-05-11
    Navicat
  • 怎么理解Javascript中浏览器对象模型
    本篇内容主要讲解“怎么理解Javascript中浏览器对象模型”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么理解Javascript中浏览器对象模型”吧!J...
    99+
    2024-04-02
  • Java和Shell有什么共同点?学习笔记和存储如何帮助你更好地理解?
    Java和Shell都是常见的编程语言,虽然它们在语法和用途上有所不同,但它们之间也有一些共同点。在本文中,我们将探讨Java和Shell的共同点,并介绍如何使用学习笔记和存储来更好地理解这些编程语言。 Java和Shell的共同点 Ja...
    99+
    2023-06-21
    shell 学习笔记 存储
  • 怎么理解并掌握mysql参数文件和类型
    这篇文章主要讲解了“怎么理解并掌握mysql参数文件和类型”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解并掌握mysql参数文件和类型”吧!1.&n...
    99+
    2024-04-02
  • 怎么解析Redis6中的单线程和多线程模型
    这篇文章的内容主要围绕怎么解析Redis6中的单线程和多线程模型进行讲述,文章内容清晰易懂,条理清晰,非常适合新手学习,值得大家去阅读。感兴趣的朋友可以跟随小编一起阅读吧。希望大家通过这篇文章有所收获!1....
    99+
    2024-04-02
  • 怎么理解Redis中的epoll和文件事件
    这篇文章主要讲解了“怎么理解Redis中的epoll和文件事件”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“怎么理解Redis中的epoll和文件事件”吧!...
    99+
    2024-04-02
  • Node.js Express 中间件的艺术:理解和掌握应用程序构建模块
    ...
    99+
    2024-04-02
  • 怎么理解多核编程中的条件同步模式
    本篇内容介绍了“怎么理解多核编程中的条件同步模式”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!在多线程编程...
    99+
    2024-04-02
  • 怎么理解Vue中的模板语法插值和指令
    本篇内容主要讲解“怎么理解Vue中的模板语法插值和指令”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“怎么理解Vue中的模板语法插值和指令”吧!Vue有很多模板语法特别好用,就是在HTML中写一些...
    99+
    2023-06-25
  • Linux系统中的/etc/passwd文件和etc/shadow文件该怎么理解
    这篇文章将为大家详细讲解有关Linux系统中的/etc/passwd文件和etc/shadow文件该怎么理解,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。在Linux系统中 /etc/pas...
    99+
    2023-06-28
  • 怎么理解C++11 中的线程及锁和条件变量
    今天就跟大家聊聊有关怎么理解C++11 中的线程及锁和条件变量,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。线程类std::thread代表一个可执行线程,使用时必须包含头文件<...
    99+
    2023-06-17
  • 怎么深入理解Java多线程与并发框中的顺序一致性模型
    怎么深入理解Java多线程与并发框中的顺序一致性模型,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。一、竞态条件(Race Condition)计算的正确性取决于 多个线程 执行...
    99+
    2023-06-05
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作