最近因为发现奇怪的 System.Data.SqlClient 性能问题(详见以前的博文),被迫提早了向 .NET Core 3.0 的升级工做(3.0 Preview 5 中问题已被修复)。郁闷的是,在刚开始对部分项目进行升级的时候就遇到了一个障碍,咱们基于 Razor Class Library 实现的自定义错误页面因为属性路由问题没法在 ASP.NET Core 3.0 Preview 5 中正常工做(详见博问),一番排查后也没找到解决方法。html
为了避免影响升级进展,咱们被迫采用了一种不经常使用的解决方法 —— 在中间件中直接调用 Controller Action 渲染视图显示自定义错误页面,也就是将原先由 ASP.NET Core Runtime 自动执行的 Controller Action (自动挡)改成手工执行(手动挡)。git
原觉得不就是比踩油门多了踩离合器和挂挡吗,应该不会很难。哪知点火后,挂挡都不知道在哪挂。Action 方法很是特殊,调用它要作不少准备工做,就如挂挡以前要先本身给车安装离合器和挂挡装置,再加上是手动挡新手,开始都不知道从哪下手。github
幸好在 ASP.NET Core 3.0 的源码中翻到了一本小册子 —— ControllerActionDescriptorBuilder.cs 中的 CreateActionDescriptor 方法,才有了点参考。浏览器
private static ControllerActionDescriptor CreateActionDescriptor(...) { var actionDescriptor = new ControllerActionDescriptor { ActionName = action.ActionName, MethodInfo = action.ActionMethod, }; actionDescriptor.ControllerName = controller.ControllerName; actionDescriptor.ControllerTypeInfo = controller.ControllerType; AddControllerPropertyDescriptors(actionDescriptor, controller); AddActionConstraints(actionDescriptor, selector); AddEndpointMetadata(actionDescriptor, selector); AddAttributeRoute(actionDescriptor, selector); AddParameterDescriptors(actionDescriptor, action); AddActionFilters(actionDescriptor, action.Filters, controller.Filters, application.Filters); AddApiExplorerInfo(actionDescriptor, application, controller, action); AddRouteValues(actionDescriptor, controller, action); AddProperties(actionDescriptor, action, controller, application); return actionDescriptor; }
在这本小手册的指导下,通过无数次熄火(NullReferenceException) 后,总算把用手动挡把车开了起来,因而有了这篇随笔分享一点驾车小经验。mvc
手动挡的操做杆主要有:RouteData, ActionDescriptor, ActionContext, ActionInvokerFactory, ControllerActionInvokerapp
其中最难操做的也是最重要的是 ActionDescriptor ,绝大多数的熄火都是在操做它时发生的,它有8个属性须要赋值,有些属性即便没用到也要进行初始化赋值,否则立马熄火(null引用异常)。异步
ActionDescriptor 的操做方法以下async
private static ActionDescriptor CreateActionDescriptor<TController>(string actionName, RouteData routeData) { var controllerType = typeof(TController); var actionDesciptor = new ControllerActionDescriptor() { ControllerName = controllerType.Name, ActionName = actionName, FilterDescriptors = new List<FilterDescriptor>(), MethodInfo = controllerType.GetMethod(actionName, BindingFlags.Public | BindingFlags.Instance), ControllerTypeInfo = controllerType.GetTypeInfo(), Parameters = new List<ParameterDescriptor>(), Properties = new Dictionary<object, object>(), BoundProperties = new List<ParameterDescriptor>() }; //... }
ControllerActionDescriptor 继承自 ActionDescriptor ,上面的赋值操做中真正传递有价值数据的是 ControllerName, ActionName, MethodInfo, ControllerTypeInfo 。一开始不知道要对哪些属性赋值,只能一步一步试,根据熄火状况一个一个添加,最终获得了上面的最少赋值操做。性能
第二重要的是 RouteData ,它是数据传输带,不只要经过它向 ActionDescriptor 传送 BindingInfo 以及向 Action 方法传递参数值,并且要向视图引擎(好比ViewEngineResult,ViewResultExecutor)传送 controller 与 action 的名称,否则视图引擎找不到视图文件。ui
RouteData 的操做方法以下
//For searching View routeData.Values.Add("controller", actionDesciptor.ControllerName.Replace("Controller", "")); routeData.Values.Add("action", actionDesciptor.ActionName); //For binding action parameters foreach (var routeValue in routeData.Values) { var parameter = new ParameterDescriptor(); parameter.Name = routeValue.Key; var attributes = new object[] { new FromRouteAttribute { Name = parameter.Name }, }; parameter.BindingInfo = BindingInfo.GetBindingInfo(attributes); parameter.ParameterType = routeValue.Value.GetType(); actionDesciptor.Parameters.Add(parameter); }
有了 ActionDescriptor 与 RouteData 以后,只需3步操做:
1)ActionContext 把离合器和挂挡装置组合起来;
2)ActionInvokerFactory 将 ActionContext 安装到车上并提供了挂挡杆 ControllerActionInvoker;
3)拉动 InvokeAsync 异步挂挡。
就能够把车开起来。
var actionContext = new ActionContext(context, routeData, actionDesciptor); var actionInvokerFactory = app.ApplicationServices.GetRequiredService<IActionInvokerFactory>(); //ActionInvokerFactory var invoker = actionInvokerFactory.CreateInvoker(actionContext); //ControllerActionInvoker await invoker.InvokeAsync();
但车没有跑在高速上,而是经过 ASP.NET Core 3.0 的 Endpoint Routing 跑在了中间件(middleware)中。
app.UseEndpoints(endpoints => { endpoints.MapGet("/", async context => { var routeData = new RouteData(); routeData.Values.Add("message", "Hello World!"); await DriveControllerAction(context, routeData, app); }); });
Contorller Action 的示例代码以下,就是将参数值传递给视图显示出来。
public class HomeController : Controller { public IActionResult Index(string message) { ViewBag.Message = message; return View(); } }
当程序一运行,浏览器请求一发出, DriveControllerAction 就开始手动挡操做,将车开起来,开车效果以下:
虽然开手动挡比自动挡麻烦不少,但驾驶时那种自主把控的感受仍是不错的,更重要的是这样的自主解决了咱们的实际问题。虽然大多数状况下都只要开自动挡,但会开手动挡会给你在解决问题时多一种选择。
完整代码见 github 上的 Startup.cs