几个核心对象:

  1. ApplicationBuilder 就是startup->Configure方法的第一个参数,请求(HttpContext) 就是由这个类来处理的
  2. HttpContext 这个就不解释了
  3. RequestDelegate 一个异步委托,委托的参数就是HttpContext,定义了一些对HttpContext的操作;可以看做是一个Action<HttpContext>(),只不过方法体内必须有异步的代码(await )

下面解释下ApplicationBuilder,这个类内部维护了一个中间件的列表,还有几个核心的方法:

  1. Use(Func<RequestDelegate, RequestDelegate> middleware),没错,就是我们常用的那个app.User(...)方法。作用是把中间件加到中间件列表
  2. Build(),构建此应用程序用于处理HTTP请求的委托。就是把HttpContext传递给中间件,由中间件处理完成后将结果返回给用户

再看看网上经典的管道图:

请求过来后,

  1. 执行中间件1的逻辑
  2. 调用Next()把处理后的HttpContext传递给中间件2
  3. 执行中间件2内的逻辑
  4. 调用Next()把HttpContext传递给中间件3
  5. 执行中间件3的逻辑
  6. 因为中间件3内没有next(),所以请求流转回中间件2
  7. 执行中间件2中next()方法后面定义的逻辑,请求流转回中间件1
  8. 执行中间件1中next()方法后的逻辑,返回结果Response。

下面是模拟的代码,因为使用了很多委托,比较烧脑,所以加了N多注释,不知道能不能说的清楚

 using System;
using System.Collections.Generic;
using System.Threading.Tasks; namespace RequestPipe
{
class Program
{
static void Main(string[] args)
{
// 实例化一个处理程序
var builder = new ApplicationBuilder(); // 把中间件加入处理程序,给HttpContext的Name加点内容
// 中间件一定要调用Next(),不然不会向后传递
// await 不解释了,当代码执行到这句后,程序会进入next()的执行流程而不会继续执行后面的语句
// 也就是说会显示 第一个开始执行,但 第一个执行结束 这句会等到 next() 运行完成后才执行
builder.Use((next) =>
async (context) =>
{
Console.WriteLine("**********第一个开始执行**********");
context.Name += "First;";
await next(context);
Console.WriteLine("**********第一个结束执行**********");
}
); builder.Use((next) =>
async (context) =>
{
Console.WriteLine("**********第二个开始执行**********");
context.Name += "Second;"; // 执行委托的方法的标准写法,也可以直接next(context)
await next.Invoke(context);
Console.WriteLine("**********第二个结束执行**********");
}
); // 特地用匿名函数来写一个,希望看起来稍微清晰一点
builder.Use(
new Func<RequestDelegate, RequestDelegate>(
delegate (RequestDelegate next)
{
return new RequestDelegate(async delegate (HttpContext context)
{
Console.WriteLine("**********第三个开始执行**********");
context.Name += "Third;";
await next(context);
Console.WriteLine("**********第三个开始执行**********");
});
}
)
); // 执行处理
builder.Build(); Console.ReadLine();
}
} public class ApplicationBuilder
{
// 中间件列表
private List<Func<RequestDelegate, RequestDelegate>> middlewares = new List<Func<RequestDelegate, RequestDelegate>>(); public void New()
{ } public void Build()
{
// 先构建一个基础的HttpContext
var baseContext = new HttpContext(); // 构建一个默认的委托(HttpContext的处理方法),就叫中间件0吧
// 如果没有经过中间件处理,就直接输出404
// 如果中间件处理成功,这里应该是输出 First;Second;Third;
var baseDelegate = new RequestDelegate(async (context) =>
{
context.Name = string.IsNullOrWhiteSpace(context.Name) ? "" : context.Name;
await context.Show();
}); // 把中间件列表的顺序反转一下
middlewares.Reverse(); // 遍历中间件列表
foreach (var middleware in middlewares)
{
// 还记得moddleware的类型吧,传入一个RequestDelegate,返回一个RequestDelegate // 经过上面的反转,现在第一个元素应该是中间件3
// baseDelegate也就是中间件0现在作为参数传递给中间件3
// 中间件3内部通过 await next(context); 保存了对默认委托的调用
// 然后将中间件3返回
// 现在 baseDelegate = 中间件3
// 接下来进入列表的第二个元素,也就是中间件2
// 和上面的逻辑一样,中间件2保存了对中间件3的引用,然后将中间件2返回出来
// ...
// 列表遍历完成后,baseDelegate = 中间件1
baseDelegate = middleware.Invoke(baseDelegate);
} // 执行中间件1
// 中间件1中保存了对中间件2的引用,所以运行到await next()的时候,就会进入到中间件2
// 同理中间件2会进入到中间件3,中间件3进入默认委托,也就是中间件0
// 中间件0执行完成(此程序中就是打印HttpContext的Name属性)返回中间件3
// 然后依次返回到中间件1,最终结束执行
baseDelegate.Invoke(baseContext);
} public void Use(Func<RequestDelegate, RequestDelegate> middleware)
{
middlewares.Add(middleware);
}
} public class HttpContext
{
public string Name { get; set; } public async Task Show()
{
Console.WriteLine(Name);
await Task.CompletedTask;
}
} public delegate Task RequestDelegate(HttpContext context); }

执行结果

**********第一个开始执行**********
**********第二个开始执行**********
**********第三个开始执行**********
First;Second;Third;
**********第三个开始执行**********
**********第二个结束执行**********
**********第一个结束执行**********

OK,这里的难点就是委托套委托,讲真的委托这东西确实强大,但代码读起来真的很难受,后面还是要整理下关于委托使用的文档,加深理解才行。

【aspnetcore】模拟中间件处理请求的管道的更多相关文章

  1. .NET Core中间件的注册和管道的构建(3) ---- 使用Map/MapWhen扩展方法

    .NET Core中间件的注册和管道的构建(3) ---- 使用Map/MapWhen扩展方法 0x00 为什么需要Map(MapWhen)扩展 如果业务逻辑比较简单的话,一条主管道就够了,确实用不到 ...

  2. .NET Core中间件的注册和管道的构建(1)---- 注册和构建原理

    .NET Core中间件的注册和管道的构建(1)---- 注册和构建原理 0x00 问题的产生 管道是.NET Core中非常关键的一个概念,很多重要的组件都以中间件的形式存在,包括权限管理.会话管理 ...

  3. WebAuthorize(中间件对所有请求进行拦截)core只有通过添加中间件过滤请求方式 而非继承然后写特性的那种方式

    一.WebAuthorize 1.项目名称 WebAuthorize 2.加个中间件 过滤请求. using Microsoft.AspNetCore.Builder; using Microsoft ...

  4. .NET Core中间件的注册和管道的构建(2)---- 用UseMiddleware扩展方法注册中间件类

    .NET Core中间件的注册和管道的构建(2)---- 用UseMiddleware扩展方法注册中间件类 0x00 为什么要引入扩展方法 有的中间件功能比较简单,有的则比较复杂,并且依赖其它组件.除 ...

  5. PHP模拟发送POST请求之五curl基本使用和多线程优化

    今天来介绍PHP模拟发送POST请求的重型武器——cURL函数库的使用和其多线程的优化方法. 说起cURL函数,可谓是老生常谈,但网上许多资料都在关键部分语焉不详,列出一大堆手册上的东西,搞得我入门时 ...

  6. PHP模拟发送POST请求之一、HTTP协议头部解析

    WEB开发中信息基本全是在POST与GET请求与响应中进行,GET因其基于URL的直观,易被我们了解,可POST请求因其信息的隐蔽,在安全的同时,也给开发者们模拟发送带来了麻烦.接下来的几篇博文中,我 ...

  7. Loadrunner模拟JSON接口请求进行测试

    Loadrunner模拟JSON接口请求进行测试     一.loadrunner脚本创建 1.Insert - New step -选择Custom Request -  web_custom_re ...

  8. WebClient模拟发送Post请求

    WebClient模拟发送Post请求方法: /// <summary> /// 模拟post请求 /// </summary> /// <param name=&quo ...

  9. httpClient模拟浏览器发请求

    一.介绍 httpClient是Apache公司的一个子项目, 用来提高高效的.最新的.功能丰富的支持http协议的客户端编程工具包.完成可以模拟浏览器发起请求行为. 二.简单使用例子 : 模拟浏览器 ...

随机推荐

  1. BZOJ 3732 Network —— 最小生成树 + 倍增LCA

    题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3732 Description 给你N个点的无向图 (1 <= N <= 15, ...

  2. discuz邮箱注册激活||腾讯企业邮箱免费注册及登录方法

    如何申请免费的企业邮箱,如果拥有了网站,还能有一个免费的域名邮箱,是不是很拉风呢?对于还没有注册企业的用户来说,优先使用企业邮箱,是非常好的事呢. 腾讯邮箱现在开放免费的企业邮箱注册,效果要比个人邮箱 ...

  3. ansible 基础知识

    英文官网,值得拥有! http://docs.ansible.com/ansible/list_of_files_modules.html# 摘自: http://blog.csdn.net/b624 ...

  4. 【MongoDB学习-在.NET中的简单操作类】

    1.新建MVC项目, 管理NuGet包,进入下载MongDB.net库文件 2.新增项目DAL数据访问层,引用以下库文件: 3.C# 访问MongoDB通用方法类: using MongoDB.Dri ...

  5. 在浏览器上直接输入 http://www.bookEstore.com就可以访问工程问题

    关于在浏览器上直接输入 http://www.bookEstore.com就可以访问工程问题 1.在tomcat/conf/server.xml文件中配置一个虚拟主机 <Host name=&q ...

  6. 微信小程序内嵌网页能力开放 小程序支持内嵌网页文档说明

    为了方便开发者灵活配置微信小程序,张小龙现在开放了小程序的内嵌网页功能,这是一个非常大的惊喜啊,以后意味着你只要开放一个手机端网站,就可以制作一个小程序了哦.操作方法1.开发者登录微信小程序后台,选择 ...

  7. vijos1842(火柴排队)

    描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度.现在将每盒中的火柴各自排成一列,同一列火柴的高度互不相同,两列火柴之间的距离定义为:∑i=1n(ai−bi)2∑i=1n(ai−bi) ...

  8. [转]解决pycharm无法导入本地包的问题(Unresolved reference 'tutorial')

    原文地址:https://www.cnblogs.com/yrqiang/archive/2016/03/20/5297519.html

  9. Docker运行MongoDB及Redis及ssh端口映射远程连接

    Docker运行MongoDB及Redis及ssh端口映射远程连接 本节内容简介 在本实验里我们将通过完成 MongoDB 和 Redis 两个容器来学习Dockerfile及Docker的运行机制. ...

  10. JS自动让手机调出软键盘,进行输入

    $('.search').click(function(){ $('input[type=text]').focus(); //让input框自动聚焦就可以让手机自动调出软键盘 });