title: Asp.Net Core底层源码剖析(一)中间件/管道
categories: 后端
tags:
- .NET

当我们像下面这样添加一个管道时发生了什么?

app.Use(async (httpcontext, next) =>
{
Console.WriteLine("做一些业务处理");
// do sth here...
// 调用下一个管道
await next();
});

接着我们来看下Use的源码,首先需要知道的是,.Net Core 源码内部ApplicationBuilder类维护了一个管道的委托调用链:

private readonly List<Func<RequestDelegate, RequestDelegate>> _components = new List<Func<RequestDelegate, RequestDelegate>>();

public delegate Task RequestDelegate(HttpContext context);

// 上面的Use先调用了这个
public static IApplicationBuilder Use(
this IApplicationBuilder app,
Func<HttpContext, Func<Task>, Task> middleware)
{
// 这里的Use调用下面的Use
return app.Use((Func<RequestDelegate, RequestDelegate>) (next => (RequestDelegate) (context =>
{
Func<Task> func = (Func<Task>) (() => next(context));
return middleware(context, func);
})));
} // 真正的Use
public IApplicationBuilder Use(
Func<RequestDelegate, RequestDelegate> middleware)
{
this._components.Add(middleware);
return (IApplicationBuilder) this;
}

从上面四段代码我们可以得到这样的信息:一个管道其实就是一个Func<RequestDelegate, RequestDelegate>类型的委托,接收一个RequestDelegate类型的委托,同时也返回一个RequestDelegate类型的委托,这个RequestDelegate委托的参数又是一个HttpContext,每次调用Use添加中间件时都会往_components这个数组的最后添加这个新的管道,下面这段是核心代码:

app.Use((Func<RequestDelegate, RequestDelegate>) (next => (RequestDelegate) (context =>
{
Func<Task> func = (Func<Task>) (() => next(context));
return middleware(context, func);
})));

这个Func<RequestDelegate, RequestDelegate>中第一个参数是入参,也就是调用下一个管道的委托,第二个参数是返回值,也就是调用当前管道的委托,这里比较难理解,我们需要分解来看:

context =>
{
Func<Task> func = (Func<Task>) (() => next(context));
return middleware(context, func);
}

这是一个RequestDelegate委托,里面做的事情就是把调用下一个管道的操作封装成立一个Func委托,然后传入当前的Func<HttpContext, Func<Task>, Task> middleware委托中,也就是我们最开始写的那一段代码,调用下一个管道的委托对应了我们自己代码中的next委托,在知道这部分的作用后,我们简化一下,记作:A

(Func<RequestDelegate, RequestDelegate>) (next => (RequestDelegate) (A))

简化之后替换一下,再看,就会发现这是一个Func<RequestDelegate, RequestDelegate>委托,第一个参数是调用下一个管道的委托,第二个是返回值,代表了调用当前管道的委托(套娃操作,这一步最难理解,需要多看几遍),这样做有什么用呢?继续往下看

最后在构建WebHostBuilder的时候,内部Build函数会利用这个来组装整个调用链:

/// <summary>
/// Produces a <see cref="T:Microsoft.AspNetCore.Http.RequestDelegate" /> that executes added middlewares.
/// </summary>
/// <returns>The <see cref="T:Microsoft.AspNetCore.Http.RequestDelegate" />.</returns>
public RequestDelegate Build()
{
RequestDelegate requestDelegate = (RequestDelegate) (context =>
{
Endpoint endpoint = context.GetEndpoint();
if (endpoint?.RequestDelegate != null)
throw new InvalidOperationException("The request reached the end of the pipeline without executing the endpoint: '" + endpoint.DisplayName + "'. Please register the EndpointMiddleware using 'IApplicationBuilder.UseEndpoints(...)' if using routing.");
context.Response.StatusCode = 404;
return Task.CompletedTask;
});
for (int index = this._components.Count - 1; index >= 0; --index)
requestDelegate = this._components[index](requestDelegate);
return requestDelegate;
}

要注意的是这里是从后往前遍历的,这里解释一下上面这段代码的作用:

  1. 首先先使用GetEndPoint()函数获取管道的终结点,可以简单理解为找出http请求最终要调用到的方法
  2. 上一步中得到了管道的最后一个位置,然后开始从后往前遍历所有的管道,并且把调用下一个管道的委托传入上一个管道作为参数

理解了上面说的话之后其实一切都已经清晰了,在委托链组装完毕后,第一个RequestDelegate内部的逻辑其实就是处理自己管道内部的业务,然后调用下一个requestDelegate(也就是next),然后一直调用下去直到到达最后的终点,或者中间没有调用next(),再总结一下:

利用上面已经组装好的委托链,当一个http请求进来的时候,.Net Core内部就会把这个http请求转换成一个HttpContext类型的参数,然后从一个管道开始处理,当一个管道处理完成后,会调用下一个委托并传入这个HttpContext继续处理相应逻辑,当然,如果你自定义的管道没有调用触发下一个管道的RequestDelegate委托,那么请求就到此中断了

我们可以发起一个请求debug调试看看是不是这样,在发起一个http请求后,我们会发现调用到了下面这段代码:

var context = application.CreateContext(this);

try
{
KestrelEventSource.Log.RequestStart(this); // 重点!!
// Run the application code for this request
await application.ProcessRequestAsync(context); // Trigger OnStarting if it hasn't been called yet and the app hasn't
// already failed. If an OnStarting callback throws we can go through
// our normal error handling in ProduceEnd.
// https://github.com/aspnet/KestrelHttpServer/issues/43
if (!HasResponseStarted && _applicationException == null && _onStarting?.Count > 0)
{
await FireOnStarting();
} if (!_connectionAborted && !VerifyResponseContentLength(out var lengthException))
{
ReportApplicationError(lengthException);
}
}

ProcessRequestAsync这个函数的调用如下:

private readonly RequestDelegate _application;

//...其他省略代码

public Task ProcessRequestAsync(HostingApplication.Context context) => this._application(context.HttpContext);

从这里我们就可以看出,_application是第一个委托,调用到这个之后很自然的就跟我们上面说得一样了,一直往下调用就完事了

总的来说,其实管道调用链组装部分其实不难理解,最难理解的是Use那段核心代码,委托套委托,每次看都会绕晕,得花很长时间才能理清里面的逻辑,如果看完之后还是觉得没看懂的话,可以自己建一个.NET 6版本的Minimal Api的Webapi项目打开源码慢慢分析,相信多花些时间肯定还是能看明白的

AspNetCore管道的更多相关文章

  1. 客官,来看看AspNetCore的身份验证吧

    开篇 这段时间潜水了太久,终于有时间可以更新一篇文章了. 通过本篇文章您将Get: Http的一些身份验证概念 在AspNetCore中实现身份验证方案 JWT等概念的基础知识 使用Bearer To ...

  2. 【aspnetcore】模拟中间件处理请求的管道

    几个核心对象: ApplicationBuilder 就是startup->Configure方法的第一个参数,请求(HttpContext) 就是由这个类来处理的 HttpContext 这个 ...

  3. ASP.NET Core HTTP 管道中的那些事儿

    前言 马上2016年就要过去了,时间可是真快啊. 上次写完 Identity 系列之后,反响还不错,所以本来打算写一个 ASP.NET Core 中间件系列的,但是中间遇到了很多事情.首先是 NPOI ...

  4. 学习ASP.NET Core, 怎能不了解请求处理管道[6]: 管道是如何随着WebHost的开启被构建出来的?

    注册的服务器和中间件共同构成了ASP.NET Core用于处理请求的管道, 这样一个管道是在我们启动作为应用宿主的WebHost时构建出来的.要深刻了解这个管道是如何被构建出来的,我们就必须对WebH ...

  5. NET Core HTTP 管道

    ASP.NET Core HTTP 管道中的那些事儿   前言 马上2016年就要过去了,时间可是真快啊. 上次写完 Identity 系列之后,反响还不错,所以本来打算写一个 ASP.NET Cor ...

  6. ASP.NET Core 源码阅读笔记(5) ---Microsoft.AspNetCore.Routing路由

    这篇随笔讲讲路由功能,主要内容在项目Microsoft.AspNetCore.Routing中,可以在GitHub上找到,Routing项目地址. 路由功能是大家都很熟悉的功能,使用起来也十分简单,从 ...

  7. ASP.NET Core 源码阅读笔记(3) ---Microsoft.AspNetCore.Hosting

    有关Hosting的基础知识 Hosting是一个非常重要,但又很难翻译成中文的概念.翻译成:寄宿,大概能勉强地传达它的意思.我们知道,有一些病毒离开了活体之后就会死亡,我们把那些活体称为病毒的宿主. ...

  8. Microsoft.AspNetCore.Routing路由

    Microsoft.AspNetCore.Routing路由 这篇随笔讲讲路由功能,主要内容在项目Microsoft.AspNetCore.Routing中,可以在GitHub上找到,Routing项 ...

  9. AspNetCore.Hosting

    Microsoft.AspNetCore.Hosting 有关Hosting的基础知识 Hosting是一个非常重要,但又很难翻译成中文的概念.翻译成:寄宿,大概能勉强地传达它的意思.我们知道,有一些 ...

  10. ASP.NET Core 2.0 : 八.图说管道

    本文通过一张GIF动图来继续聊一下ASP.NET Core的请求处理管道,从管道的配置.构建以及请求处理流程等方面做一下详细的研究.(ASP.NET Core系列目录) 一.概述 上文说到,请求是经过 ...

随机推荐

  1. doecker---制作DockerFile并上传Hub

    一.DockerFile基础知识 FROM #基础镜像,一切从这里开始构建 MAINTAINER #镜像是谁写的,姓名+邮箱 RUN #镜像构建的时候需要运行的命令 ADD #添加内容,步骤,tomc ...

  2. java简易两数计算器

    public class calculator { public static void main(String[] args) { Scanner scanner = new Scanner(Sys ...

  3. ES6 学习笔记(十)Map的基本用法

    1 基本用法 Map类型是键值对的有序列表,而键和值都可以是任意类型.可以看做Python中的字典(Dictionary)类型. 1.1 创建方法 Map本身是一个构造函数,用来生成Map实例,如: ...

  4. salesforce零基础学习(一百二十)快去迁移你的代码中的 Alert / Confirm 以及 Prompt吧

    本篇参考: https://developer.salesforce.com/blogs/2022/01/preparing-your-components-for-the-removal-of-al ...

  5. Django系列---开发三 前后端分离

    数据交互接口规范REST,全称 Representational State Transfer,意为"表现层状态转化". django的第三方拓展--django-rest-fra ...

  6. 「浙江理工大学ACM入队200题系列」问题 L: 零基础学C/C++52——计算数列和2/1,3/2,5/3,8/5......

    本题是浙江理工大学ACM入队200题第五套中的L题 我们先来看一下这题的题面. 题面 题目描述 有一分数序列:2/1,3/2,5/3,8/5,13/8,21/13,-- 计算这个数列的前n项和.注意: ...

  7. 浅谈MYSQL的索引以及它的数据结构

    什么是索引 mysql的数据是持久化到磁盘的,写SQL查询数据也就是在磁盘的某个位置查找符合条件的数据,但是磁盘IO比起内存效率是极慢的,特别是数据量大的时候,这时候就需要引入索引来提高查询效率: 在 ...

  8. iview table json数据里的num排序问题

    title: 'Num', key: 'num', sortable: true, sortMethod:function(a,b,type){ //可以用Number()或者parseInt(a)转 ...

  9. Conda 环境移植 (两种方式)

    ------------------------方法一------------------------ 优点: 在原机器上需要进行的操作较少,且除了conda不需要其余的库来支撑:需要传输的文件小,操 ...

  10. jsp 页面返回、本页面刷新

    返回上一页面: window.history.go(-1);  //返回上一页window.history.back();  //返回上一页 返回上一页面并对上一页面刷新: history.go(-1 ...