目录:

1、Core 处理HTTP请求流程

2、中间件(Middleware)&&处理流程

3、创建自定义中间件&&模拟Core的请求管道

Core 处理HTTP请求流程

Http请求来到之后,会先由WebServer( 例如IIS,Nginx) 拿到然后丢给Kestrel WebServer,再有Kestrel 将Http请求封装为HttpContext后传入给.net core 请求管道(RequestDelegate)。HttpContext经过一系列middleware后返回response给WebServer。

中间件(Middleware)&&处理流程

  微软官方文档给出的解释是这样的:中间件是加载在应用程序管道中以处理和相应请求的组件。其中每个组件都可以 选择是否可以将请求传递给管道中下一个组件;可以在调用管道的下一个组件之前或之后执行。下面是官方文档给出的一张图。

  也就是说一个http请求的处理过程为,经过中间件1的处理逻辑(这个时候可以在中间件1中对httpcontext做一些处理,例如写入一段内容),调用next()方法,进入到中间件2中,重复以上步骤。最终,所有中间件的处理逻辑的输出结果都会叠加到HttpResponse中去。

  .net core 中有三种方式调用中间件。在StartUp的Configure方法中,分别可以调用app.Use(),app.Run(),app.Map() 来调用中间件。接下来让我们来看下:

  app.Use方法有两个重载形式:

第一个方法接收一个中间件,第二个方法接收一个上下文和一个中间件。让我们来分别调用下:

             app.Use(next =>
{
return context =>
{
context.Response.WriteAsync("this is middleware1 start... ");
return next(context);
};
});
app.Use(async (context, next) =>
{
await context.Response.WriteAsync("this is middleware2 start... ");
await next();
});

  app.Run方法会终止管道,返回HttpResponse。在我们Configure方法中,默认最后一个方法就是app.Run 输出一句HelloWorld ,如下图

  Use和Run都比较简单,不再详细介绍了。来看一眼程序运行结果:

  app.Map 方法用来构建分支管线,它会启用一个新的ApplicationBuilder。Map根据请求的路径来匹配分支管线。Map方法也可以根据不同的PathString来匹配多个分支管线或嵌套匹配。

            app.Map("/task", build =>
{
build.Run(async (context) =>
{
await context.Response.WriteAsync("this is middleware3 For Map start... ");
});
});
             app.Map("/map", build =>
{
build.Map("/mapTest2", MapHandleTest2);
build.Run(async (context) =>
{
await context.Response.WriteAsync("this is middleware3 For Map start... ");
});
});
app.Map("/mapTest1", MapHandleTest1);
         private void MapHandleTest1(IApplicationBuilder applicationBuilder)
{
applicationBuilder.Run(async context =>
{
await context.Response.WriteAsync("this is middleware3HandleTest1 For Map start... ");
});
}
private void MapHandleTest2(IApplicationBuilder applicationBuilder)
{
applicationBuilder.Run(async context =>
{
await context.Response.WriteAsync("this is middleware3HandleTest2 For Map start... ");
});
}

  Map还有个特殊的方法,MapWhen, 它接收一个Func<HttpContext, bool>的参数,可以根据Http请求上下文做一些逻辑判断。然后继续执行管道

             app.MapWhen(context => context.Request.Query.ContainsKey("branch"),
build =>
build.Run(async context =>
{
var branchVer = context.Request.Query["branch"];
await context.Response.WriteAsync($"Branch used = {branchVer}");
})
);

,这里再贴张.net core 自带的中间件

  

创建自定义中间件&&模拟Core的请求管道

  管道的实现机制由ApplicationBuilderRequestDelegate 组成。RequestDelegate是Middleware的核心,Use方法会将RequestDelegate加入到ApplicationBuilder 中去,最后执行Build方法,把所有RequestDelegate reverse之后挨个执行一遍。

  接下来编写一个自定义的Middleware。

    public class RequestTestMiddleware
{
private readonly RequestDelegate _next; //中间件核心RequestDelegate,_next 表示执行下一个中间件。
public RequestTestMiddleware(RequestDelegate next)
{
_next = next;
}
public Task Invoke(HttpContext context)
{
context.Response.WriteAsync("hello this is my RequestTestMiddleware");
return _next(context); //覆盖delegate 的Inboke方法,将http上下文传入下一个中间件
}
} public static class RequestTestMiddlewareExtensions
{
public static IApplicationBuilder UseRequestTest(this IApplicationBuilder builder) //使用扩展方法使我们的中间件对外开放。
{
return builder.UseMiddleware<RequestTestMiddleware>(); //将我们中间件加入到IApplicationBuilder中去。
}
}
  //在StartUp 中使用我们的自定义中间件。
app.UseRequestTest();

Http请求管道的模拟构建

     /// <summary>
/// 拿到请求上下文,返回一个委托
/// </summary>
/// <param name="contex"></param>
/// <returns></returns>
public delegate Task RequestDelegate(Context contex);
/// <summary>
/// 模拟http请求上下文
/// </summary>
public class Context
{ }

RequestDelegate和Context

         public static IList<Func<RequestDelegate, RequestDelegate>> _list = new
List<Func<RequestDelegate, RequestDelegate>>();
static void Main(string[] args)
{
Use(next => //加入中间件1
{
return context =>
{
Console.WriteLine("this is use middleware1");
return next.Invoke(context);
};
});
Use(next => //加入中间件2
{
return context =>
{
Console.WriteLine("this is use middleware2");
return next.Invoke(context);
};
});
RequestDelegate end = context =>
{
Console.WriteLine("this is end..."); //管道
return Task.CompletedTask;
};
_list.Reverse();
foreach (var middleware in _list)
{
end = middleware.Invoke(end);
}
end.Invoke(new Context()); //实例化一个上下文实例
Console.ReadLine();
}
/// <summary>
/// 将middleware 加入到list(applicationbuilder)中
/// </summary>
public static void Use(Func<RequestDelegate, RequestDelegate> iMiddleware)
{
_list.Add(iMiddleware);
}

Use middleware和管道执行

Jesse博客学习笔记。传送门=》 http://video.jessetalk.cn/

Core篇——初探Core的Http请求管道&&Middleware的更多相关文章

  1. Core篇——初探Core的认证,授权机制

    目录 1.Cookie-based认证的实现 2.Jwt Token 的认证与授权 3.Identity Authentication + EF 的认证 Cookie-based认证的实现 cooki ...

  2. Core篇——初探Core配置管理

    文章目录 1.命令行配置 2.Json文件配置 3.配置文件文本至C#对象实例的映射 4.配置文件热更新 5.总结 命令行的配置 我们首先来创建一个.net core 的控制台项目,然后引入.net ...

  3. Core篇——初探IdentityServer4(OpenID Connect模式)

    Core篇——初探IdentityServer4(OpenID Connect客户端验证) 目录 1.Oauth2协议授权码模式介绍2.IdentityServer4的OpenID Connect客户 ...

  4. Core篇——初探IdentityServer4(客户端模式,密码模式)

    Core篇——初探IdentityServer4(客户端模式,密码模式) 目录 1.Oatuth2协议的客户端模式介绍2.IdentityServer4客户端模式实现3.Oatuth2协议的密码模式介 ...

  5. Core篇——初探依赖注入

    目录 1.DI&&IOC简单介绍 2.UML类图中六种关联关系 3..net core 中DI的使用 4..net core DI初始化源码初窥 DI&&IOC简单介绍 ...

  6. 比较ASP.NET和ASP.NET Core[经典 Asp.Net v和 Asp.Net Core (Asp.Net Core MVC)]

    ASP.NET Core是.与.Net Core FrameWork一起发布的ASP.NET 新版本,最初被称为ASP.NET vNext,有一系列的命名变化,ASP.NET 5.0,ASP.NET ...

  7. ASP.NET Core Blazor 初探之 Blazor Server

    上周初步对Blazor WebAssembly进行了初步的探索(ASP.NET Core Blazor 初探之 Blazor WebAssembly).这次来看看Blazor Server该怎么玩. ...

  8. ASP.NET Core 运行原理解剖[3]:Middleware-请求管道的构成

    在 ASP.NET 中,我们知道,它有一个面向切面的请求管道,有19个主要的事件构成,能够让我们进行灵活的扩展.通常是在 web.config 中通过注册 HttpModule 来实现对请求管道事件监 ...

  9. Net Core 的公共组件之 Http 请求客户端

    Net Core 的公共组件之 Http 请求客户端 想必大家在项目开发的时候应该都在程序中调用过自己内部的接口或者使用过第三方提供的接口,咱今天不讨论 REST ,最常用的请求应该就是 GET 和 ...

随机推荐

  1. Spring学习笔记之基础、IOC、DI(1)

    0.0 Spring基本特性 Spring是一个开源框架:是基于Core来架构多层JavaEE系统 1.0 IOC 控制反转:把对象的创建过程交给spring容器来做. 1.1 application ...

  2. 查看tomcat的内存情况

    查看现有tomcat的内存大小情况 1.启动tomcat 2.访问 http://localhost:8080/manager/status ,并输入您在安装tomcat时输入的用户与口令,如 adm ...

  3. bootstrap3的 progress 进度条

    : 2.3版               3.0版 .bar .progress-bar .bar-* .progress-bar-* 2.代码: <!DOCTYPE html PUBLIC & ...

  4. 前端自动化构建yeoman

    前端自动化可分为: yo(脚手架工具).grunt(构建工具).bower(包管理器). OMAN的特性 http://yeoman.io/learning/   闪电般的初始化:项目开始阶段,可以基 ...

  5. 洛谷P2822 组合数问题 杨辉三角

    没想到这道题竟然这么水- 我们发现m,n都非常小,完全可以O(nm)O(nm)O(nm)预处理出stripe数组,即代表(i,j)(i,j)(i,j) 及其向上的一列的个数,然后进行递推即可. #in ...

  6. 新手学python-Day1-鸡汤,变量,编码,进制,判断

    作业要求为编写一个登录接口:`输入用户名密码`认证成功后显示欢迎信息`输错三次锁定做完作业以后,使用ctrl+alt+f格式化一下,特别舒爽 1 def jilu(str): 2 zt = open( ...

  7. Vue -- element-ui el-table 的合计在第一行显示并可点击

    使用element-ui el-table 中有这样一个需求,需要将合计放在表格内容的第一行,并且点击合计可跳转到其它页面! 框架中提供了合计的属性方法,这样可以进行数值求和及自定义求和,但是,合计那 ...

  8. 安装oracle服务端之后再安装oracle客户端导致sqlplus命令无法使用??

    首先小编自述一下所遇到的问题: 昨天在已经安装 oracle 服务端的 win7 X64 主机上安装 oracle client 之后,发现我的sqlplus命令无法用了??经过百度,都说我服务没有开 ...

  9. FreeMarker 语法 list

    一.java 代码 @Test public void testFreeMarker() throws Exception { //1.创建一个模板文件 //2.创建一个Configuration对象 ...

  10. Redis-Cluster集群原理

    一.redis-cluster 官方推荐的 redis 集群解决方案,优点在于去中心化, 去中间件,也就是说,集群中的每个节点都是平等的关系,都是对等的,每个节点都保存各自的数据和整个集群的状态.每个 ...