http://www.javashuo.com/article/p-xshwgqda-bb.htmlhtml
在上篇文章主要介绍了DotNetCore项目情况,本篇文章是咱们在开发本身的项目中实际使用的,比较贴合实际应用,算是对中间件的一个深刻使用了,不是简单的Hello World,若是你以为本篇文章对你有用的话,不妨点个【推荐】。web
咱们知道,任何的一个web框架都是把http请求封装成一个管道,每一次的请求都是通过管道的一系列操做,最终到达咱们写的代码中。那么中间件就是在应用程序管道中的一个组件,用来拦截请求过程进行一些其余处理和响应。中间件能够有不少个,每个中间件均可以对管道中的请求进行拦截,它能够决定是否将请求转移给下一个中间件。app
asp.net core 提供了IApplicationBuilder
接口来让把中间件注册到asp.net的管道请求当中去,中间件是一个典型的AOP应用。 下面是一个微软官方的一个中间件管道请求图:框架
能够看到,每个中间件都均可以在请求以前和以后进行操做。请求处理完成以后传递给下一个请求。asp.net
默认状况下,中间件的执行顺序根据Startup.cs
文件中,在public void Configure(IApplicationBuilder app){}
方法中注册的前后顺序执行。
大概有3种方式能够在管道中注册"中间件"ui
app.Use()
,IApplicationBuilder
接口原生提供,注册等都用它。 app.Run()
,是一个扩展方法,它须要一个RequestDelegate
委托,里面包含了Http的上下文信息,没有next参数,由于它老是在管道最后一步执行。 app.Map()
,也是一个扩展方法,相似于MVC的路由,用途通常是一些特殊请求路径的处理。如:www.example.com/token 等。上面的Run,Map内部也是调用的Use,算是对IApplicationBuilder接口扩充,若是你以为名字都不够准确,那么下面这个扩展方法就是正宗的注册中间件的了,也是功能最强大的。app.UseMiddleware<>()
,没错,就是这个了。 为何说功能强大呢?是由于它不但提供了注册中间件的功能,还提供了依赖注入(DI)的功能,之后大部分状况就用它了。this
熟悉MVC框架的同窗应该知道,MVC也提供了5大过滤器供咱们用来处理请求先后须要执行的代码。分别是AuthenticationFilter
,AuthorizationFilter
,ActionFilter
,ExceptionFilter
,ResultFilter
。.net
根据描述,能够看出中间件和过滤器的功能相似,那么他们有什么区别?为何又要搞一个中间件呢?
其实,过滤器和中间件他们的关注点是不同的,也就是说职责不同,干的事情就不同。日志
举个栗子,中间件像是
埃辛诺斯战刃
,过滤器像是巨龙之怒,泰蕾苟萨的寄魂杖
,你一个战士拿着巨龙之怒,泰蕾苟萨的寄魂杖
去战场杀人,虽然都有伤害,可是你拿着法杖伤害低不说,还减属性啊。code
同做为两个AOP利器,过滤器更贴合业务,它关注于应用程序自己,好比你看ActionFilter
和 ResultFilter
,它都直接和你的Action,ActionResult交互了,是否是离你很近的感受,那我有一些好比对个人输出结果进行格式化啦,对个人请求的ViewModel进行数据验证啦,确定就是用Filter无疑了。它是MVC的一部分,它能够拦截到你Action上下文的一些信息,而中间件是没有这个能力的。
那么,什么时候使用中间件呢?个人理解是在咱们的应用程序当中和业务关系不大的一些须要在管道中作的事情可使用,好比身份验证,Session存储,日志记录等。其实咱们的 asp.net core项目中自己已经包含了不少个中间件。
举例,咱们在新建一个 asp.net core应用程序的时候,默认生成的模板当中
public void Configure(IApplicationBuilder app, ILoggerFactory loggerFactory) { app.UseDeveloperExceptionPage(); app.UseStaticFiles(); loggerFactory.AddConsole(); app.UseMvc(routes => { routes.MapRoute( name: "default", template: "{controller=Home}/{action=Index}/{id?}"); }); }
懒得去下载源码了,咱们使用Reflector去查看源码:
//扩展方法`app.UseDeveloperExceptionPage();` public static class DeveloperExceptionPageExtensions { // Methods public static IApplicationBuilder UseDeveloperExceptionPage(this IApplicationBuilder app) { if (app == null) { throw new ArgumentNullException("app"); } return UseMiddlewareExtensions.UseMiddleware<DeveloperExceptionPageMiddleware>(app, Array.Empty<object>()); } }
//扩展方法`app.UseStaticFiles();` public static class StaticFileExtensions { // Methods public static IApplicationBuilder UseStaticFiles(this IApplicationBuilder app) { if (app == null) { throw new ArgumentNullException("app"); } return UseMiddlewareExtensions.UseMiddleware<StaticFileMiddleware>(app, Array.Empty<object>()); } }
能够看到 app.UseDeveloperExceptionPage()
,app.UseStaticFiles()
等等都是经过中间件实现的。