前言
ASP.NET Core的Middleware(中间件)就是使用了管道模式:
Request(请求)在管道中传递,依次经过管道中的每一个MiddleWare进行处理。
MiddleWare就像一层层的“滤网”,过滤所有的请求和响应。
这种模式的好处在于,分离关注点。比如验证用户、记录访问日志,可以将这些任务分解到不同MiddleWare中,互相解耦。
既然ASP.NET Core已经实现了管道模式,为什么还要在业务层实现一遍呢?
原因
我觉得可以分为2方面考虑:
与顶级框架解耦,不能保证应用程序代码始终运行在ASP.NET Core下,如果将业务迁移到WPF,记录访问日志这类放在管道中的功能还是需要实现
同时支持不同来源请求,请求可能来自于Web API,也可能来自于Windows Service定时调用,但业务层处理逻辑应保持一样
因此,ASP.NET Core框架应该只用于接收输入和返回输出,而在业务层实现管道模式。
定义API的最佳实践
在《定义API的最佳实践MediatR
类库实现Controller方法,将业务逻辑和Controller进行隔离。
示例代码如下:
[HttpGet]
public async Task<string> Demo([FromQuery] DemoQuery request)
{return await _mediator.Send(request);
}public class DemoQuery : IRequest<string>
{public string Name { get; set; }
}public class DemoQueryHandler : IRequestHandler<DemoQuery, string>
{ public async Task<string> Handle(DemoQuery request, CancellationToken cancellationToken){Console.WriteLine("DemoQueryHandler执行");return request.Name;}
}
而MediatR不仅仅是一个简单的中介模式实现,它还提供了Behaviors
的概念:
Behaviors非常类似于ASP.NET Core中的MiddleWare,可以让我们实现管道模式。
管道实现
要定义Behaviors,我们需要实现IPipelineBehavior
接口,示例代码如下:
public class FirstPipelineBehavior<TRequest, TResponse>: IPipelineBehavior<TRequest, TResponse>
{public async Task<TResponse> Handle(TRequest request, CancellationToken cancellationToken, RequestHandlerDelegate<TResponse> next){Console.WriteLine("FirstPipelineBehavior执行中");var response = await next();Console.WriteLine("FirstPipelineBehavior执行完成");return response;}
}
和MiddleWare类似,调用next()
可以将请求向下传递。
接着在Startup.cs
中注册管道:
services.AddMediatR(typeof(Startup));
services.AddTransient(typeof(IPipelineBehavior<,>), typeof(FirstPipelineBehavior<,>));
services.AddTransient(typeof(IPipelineBehavior<,>), typeof(SecondPipelineBehavior<,>));
services.AddTransient(typeof(IPipelineBehavior<,>), typeof(ThirdPipelineBehavior<,>));
访问API可以看到,管道可以在IRequestHandler
执行之前和之后执行代码,并且处理顺序和注册顺序相同:
结论
通过本文,我们了解到,通过MediatR提供的Behaviors功能,不依赖ASP.NET Core框架,可以在业务层实现管道模式。
如果你觉得这篇文章对你有所启发,请关注我的个人公众号”My IO“,记住我!