7

我有一个最近从 ASP.NET MVC1 升级到 ASP.NET MVC4 rc1 的应用程序。

它使用 Webforms 视图引擎。

每当使用 Url.Action(action,controller) 时,它都会出现性能问题。

我可以在 ASP.NET MVC3 中重现该问题。

我需要 3 毫秒来呈现在 ASP.NET MVC1 中包含 10 个 Url.Action 帮助器实例的视图,并且需要 40 毫秒来在 ASP.NET MVC3 中呈现相同的视图。

我已经找到了一些让它渲染得更快的方法:

  • 我将默认路由移到顶部

  • 我删除了 Url.Action 并使用了静态链接

这感觉不对:应用程序非常大,我需要其中的体面工作路由的优点。我也不相信我发现了所有的性能瓶颈。路由是 MVC 的核心部分:如果有什么表现不好,它会在应用程序的不同部分弹出。

我的印象是 MVC3 引入了一些路由功能(如正则表达式约束),即使我不使用它们也会导致应用程序性能不佳。

有什么我可以做的,比如打开路由功能或使用一组不同的 URL 助手?

此代码重现了该问题:

索引操作

public ActionResult Index()
        {

            return View();
        }

索引.aspx

<%@ Page Language="C#"  Inherits="System.Web.Mvc.ViewPage" %>

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head >
    <title></title>
    <link href="../../Content/Site.css" rel="stylesheet" type="text/css" />
</head>

<body>
    <div class="page">
<%= Url.Action("Action1", "Controller1") %>
<%= Url.Action("Action2", "Controller2") %>
<%= Url.Action("Action3", "Controller3") %>
<%= Url.Action("Action4", "Controller4") %>
<%= Url.Action("Action5", "Controller5") %>
<%= Url.Action("Action6", "Controller6") %>
<%= Url.Action("Action7", "Controller7") %>
<%= Url.Action("Action8", "Controller8") %>
<%= Url.Action("Action9", "Controller9") %>
<%= Url.Action("Action10", "Controller10") %>
    </div>
</body>
</html>

路由注册 这看起来很奇怪:但我只是想模拟我的不是很复杂的路由。这不是SO的600条路线!

public static void RegisterRoutesSlow(RouteCollection routes)
{
    routes.IgnoreRoute("{resource}.axd/{*pathInfo}");
    routes.IgnoreRoute("{language}/Content/{*pathInfo}");

    routes.IgnoreRoute("images/{*pathinfo}");
    routes.IgnoreRoute("scripts/{*pathinfo}");
    routes.IgnoreRoute("content/{*pathinfo}");
    routes.IgnoreRoute("{file}.gif");
    routes.IgnoreRoute("{file}.jpg");
    routes.IgnoreRoute("{file}.js");
    routes.IgnoreRoute("{file}.css");
    routes.IgnoreRoute("{file}.png");
    routes.IgnoreRoute("{file}.pdf");
    routes.IgnoreRoute("{file}.htm");
    routes.IgnoreRoute("{file}.html");
    routes.IgnoreRoute("{file}.swf");
    routes.IgnoreRoute("{file}.txt");
    routes.IgnoreRoute("{file}.xml");
    routes.IgnoreRoute("{*favicon}", new { favicon = @"(.*/)?favicon.ico(/.*)?" });

    for (int i = 0; i <= 10; i++)
    {
        routes.MapRoute(
            // Route name
            "RouteName" + i.ToString(),
            // URL with parameters                              
            "{language}/{controller}/{action}/{para1}",
            // Parameter defaults
            new
            {
                action = "Index",
                language = "de",
                para1 = 0
            },
            //Parameter constraints
            new { language = "de|en", controller = "SomeNameOfAnActualController" + i.ToString() }
            );
    }
    routes.MapRoute(
                   "DefaulRoute",            // Route name
                   "{controller}/{action}",    // URL with parameters
                   new
                   {
                       controller = "Home",
                       action = "Index",
                   }
               );
    routes.MapRoute("404-PageNotFound", "{*url}", new { controller = "Error", action = "PageNotFound", language = "de" });
}

编辑

示例代码现在是针对 MVC2 编译的。在 VS2010 中,MVC2 可以针对 .NET 3.5 或 4.0 进行编译。

3.5 的性能很好,4.0 的性能很差。

我猜这意味着性能不佳的部分不在 MVC 程序集中,而是在框架程序集中(如 System.Web.Routing.dll)。问题还是一样:我能做点什么吗?一个可接受的答案也是:不,代码很慢,因为从 3.5 版到 4.0 版 MS 更改了 XXX

编辑-2

我反编译了 System.Web.Routing.dll 需要很长时间的部分。它使用编译的正则表达式。有一个代码路径(constraint2.Match)在不执行正则表达式的情况下返回,但我还没有检查它是否在内部使用了不同的昂贵操作。

protected virtual bool ProcessConstraint(HttpContextBase httpContext, object constraint, string parameterName, RouteValueDictionary values, RouteDirection routeDirection)
{
    object obj2;
    IRouteConstraint constraint2 = constraint as IRouteConstraint;
    if (constraint2 != null)
    {
        return constraint2.Match(httpContext, this, parameterName, values, routeDirection);
    }
    string str = constraint as string;
    if (str == null)
    {
        throw new InvalidOperationException(string.Format(CultureInfo.CurrentUICulture, SR.GetString("Route_ValidationMustBeStringOrCustomConstraint"), new object[] { parameterName, this.Url }));
    }
    values.TryGetValue(parameterName, out obj2);
    string input = Convert.ToString(obj2, CultureInfo.InvariantCulture);
    string pattern = "^(" + str + ")$";
    return Regex.IsMatch(input, pattern, RegexOptions.CultureInvariant | RegexOptions.Compiled | RegexOptions.IgnoreCase);
}
4

3 回答 3

3

解决了与您类似的问题:在页面上第一次调用 Url.Action 很慢 关于路由约束的结论与正则表达式约束非常慢。

于 2012-08-17T21:56:43.667 回答
0

每个视图在第一次使用时都会被编译和缓存。但是,由于 aspx 视图不是专门为 Mvc 设计的,因此每个 Url.Action 不会在最终链接中全部编译一次,而是在每次执行时重新计算。Razor 编译器有更好的优化。唯一的解决方案是使用 Url.Action 计算各种链接并将它们存储到某个应用程序级别的属性中,因此它仅在第一次执行时计算。您可以将它们放在应用程序字典或类的静态属性中。

于 2012-08-14T08:17:56.177 回答
0

我不确定您所看到的原因,但它可能不仅是 MVC 1 与 MVC 4,更高版本中的 IIS 设置会影响视图渲染的速度。几个月前,我遇到了一个幻灯片,我认为它非常有趣,与 MVC 3 应用程序中的性能改进技巧有关。

http://www.slideshare.net/ardalis/improving-aspnet-mvc-application-performance

具体来说,请看幻灯片 28,其中指出:

卸载 IIS UrlRewrite 模块

  • 如果服务器上没有应用程序正在使用它
  • 在 v3 之前的 MVC 应用程序中无效
  • 提高 URL 生成速度

我认为这意味着 UrlRewrite 模块将对 MVC 3 产生负面影响,但不会对 MVC 2 或 1 产生负面影响,这可能是您所看到的减速的根源。还有其他一些改进,但我不相信它们中的任何一个“直接”与您所看到的有关。

于 2012-08-14T13:56:23.867 回答