如何从 url 或路由数据中获取控制器类型和操作信息?

Posted

技术标签:

【中文标题】如何从 url 或路由数据中获取控制器类型和操作信息?【英文标题】:How can I get controller type and action info from a url or from route data? 【发布时间】:2011-02-11 01:23:01 【问题描述】:

在给定System.Web.Routing.RouteData 的情况下,如何获取将要调用的控制器操作(方法)和控制器类型?

我的场景是这样的 - 我希望能够在 OnActionExecuting 方法中执行某些操作(或不执行)。

然而,我经常想知道的不是当前动作,而是被调用的“根”动作;我的意思是我可能有一个名为“登录”的视图,这是我的登录页面。此视图可能包括 另一个局部视图“LeftNav”。当为 LeftNav 调用 OnActionExecuting 时,我希望能够确定它确实是为 Login 的“根”动作调用的。

我意识到通过调用RouteTable.Routes.GetRouteData(actionExecutingContext.HttpContext),我可以获得“root”请求的路由,但是如何将其变成 方法和类型信息?

到目前为止,我唯一的解决方案是:

 var routeData = RouteTable.Routes.GetRouteData(actionExecutingContext.HttpContext)
 var routeController = (string)routeData.Values["controller"]; 
 var routeAction = (string)routeData.Values["action"];

这样做的问题是“routeController”是去掉了“Controller”后缀的控制器名称,并且不是完全限定的;即它是“登录”,而不是“MyCode.Website.LoginController”。

如果可能的话,我宁愿得到一个实际的 TypeMethodInfo,或者至少是一个完全限定的类型名称。

有什么想法或替代方法吗?

[编辑 - 这是 ASP.Net MVC 1.0]

【问题讨论】:

【参考方案1】:
  protected override void OnActionExecuting(ActionExecutingContext filterContext)
  
     var type1 = filterContext.Controller.GetType();
     var type2 = filterContext.ActionDescriptor
                    .ControllerDescriptor.ControllerType;
  

好的,抱歉,我错过了“根”部分。

然后,另一种方式,您可以将控制器类型保存到线程存储中。伪代码:

  protected override void OnActionExecuting(ActionExecutingContext filterContext)
  
     if (!Thread.LocalStorage.Contains("root_controller"))
        Thread.LocalStorage["root_controller"] = 
            filterContext.ActionDescriptor
                    .ControllerDescriptor.ControllerType;
  

只是一个想法。我确定线程本地存储在 C# 中可用。这里的关键思想是您只为第一次请求保存它,因此它始终是根控制器。

【讨论】:

不幸的是,这不起作用,因为它返回 current 控制器动作的类型,而不是我给出的示例中的“根”控制器动作。我之所以调用 GetRouteData(actionExecutingContext.HttpContext) 是因为这确实成功地为我提供了“根”的路由,但我无法将其转换为控制器类型和方法。 不要在 ASP.NET 应用程序中使用 ThreadLocal 存储。 ASP.NET 请求可以在线程之间来回跳转,因此 ThreadLocal 存储可能会在您意想不到的时候消失。如果您需要存储仅为当前请求保留的信息,请改用 HttpContext.Items。 实际上我只是不知道有任何 ASP.NET 集合仅存储当前请求的数据。当然它比线程存储更好。感谢您的提示。【参考方案2】:

这是我从各种来源编译的解决方案。 url 变量应包含操作的 URL:

        url = "YOUR URL";
        // Original path is stored and will be rewritten in the end
        var httpContext = new HttpContextWrapper(HttpContext.Current);
        string originalPath = httpContext.Request.Path;

        try
        
            // Fake a request to the supplied URL into the routing system
            httpContext.RewritePath(url);
            RouteData urlRouteData = RouteTable.Routes.GetRouteData(httpContext);

            // If the route data was not found (e.g url leads to another site) then authorization is denied.
            // If you want to have a navigation to a different site, don't use AuthorizationMenu
            if(urlRouteData != null)
            
                string controllerName = urlRouteData.Values["controller"].ToString();
                string actionName = urlRouteData.Values["action"].ToString();

                // Get an instance of the controller that would handle this route
                var requestContext = new RequestContext(httpContext, urlRouteData);
                var controllerFactory = ControllerBuilder.Current.GetControllerFactory();
                var controller = (ControllerBase) controllerFactory.CreateController(requestContext, controllerName);

                // Find the action descriptor
                var controllerContext = new ControllerContext(httpContext, new RouteData(), controller);
                var controllerDescriptor = new ReflectedControllerDescriptor(controller.GetType());
                var actionDescriptor = controllerDescriptor.FindAction(controllerContext, actionName);
            
        
        finally
        
            // Reset our request path.
            httpContext.RewritePath(originalPath);
        

【讨论】:

【参考方案3】:
public Type ControllerType(string controllerName)

   var fullName = controllerName + "Controller";
   var assemblyName = Assembly.GetExecutingAssembly().FullName;
   return Activator.CreateInstance(assemblyName, fullTypeName).GetType();


public MethodInfo ActionMethodInfo(string actionName, Type controllerType)

   return controllerType.GetMethod(actionName);

您是否正在考虑类似的实现?需要一些 Try/Catches!

【讨论】:

谢谢丹 - 这有点像我要去的地方,但真的希望避免。在某个地方,MVC 确切地知道它将调用什么方法、什么类型。这种方法(和我的)的问题是我必须附加后缀“Controller”,并且还必须将命名空间添加到类型等之前。它可能会工作,但如果我可以挂钩MVC 已经在做的事情,我宁愿那样做。 GetMethod() 将抛出 AmbiguousMatchException 如果有多个同名方法,这是一种非常常见的模式。通常,您还必须考虑与 RouteData 中的其他值相关的匹配项,以便选择正确的操作方法。【参考方案4】:

MvcSiteMapProvider 执行此操作。这是这个特殊事物的代码。

Here is the code

【讨论】:

如何定位Action方法?

以上是关于如何从 url 或路由数据中获取控制器类型和操作信息?的主要内容,如果未能解决你的问题,请参考以下文章

Asp.Net MVC:如何获取当前控制器/视图的虚拟 url?

从 url 获取路由

如何使用 c# asp.net 从 url 获取数据?

无法从角度路由器获取路径或 url

Zend Framework 2 中的路由,跳过 url 中的“索引”操作但获取 id

Zend Framework:从路由获取子域参数