MVC RouteConfig "CatchAll" {*url} 不工作

MVC RouteConfig "CatchAll" {*url} not working

我正在尝试 "catch all" 我的 MVC 应用程序中的 500 和 404 错误,我似乎无法掌握需要什么,即使在阅读了所有文章和问题之后也是如此。

Web.config(这允许 500 个错误转到 ~/Views/Shared/Error.cshtml):

<system.web>
    <customErrors mode="On" redirectMode="ResponseRewrite" />
</system.web>

我已将 HomeController 设置为抛出错误以测试上述设置:

public ActionResult Index()
{
    //Testing errors
    throw new Exception("Exception");

    return View();
}

在我的 Global.asax.cs 中,我有以下记录 500 个错误:

protected void Application_Error()
{
    var ex = Server.GetLastError();

    //Custom ExceptionLog
    new ExceptionLogHelper().Add("Application_Error", Response.Status, ex);
}

现在针对 404 错误:

在我的 RouteConfig.cs 中,我有以下路线,但似乎无法捕获所有 404:

public class RouteConfig
{
    public static void RegisterRoutes(RouteCollection routes)
    {
        routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

        routes.MapRoute(
            name: "Error",
            url: "Error/{code}",
            defaults: new { controller = "Error", action = "Index", code = UrlParameter.Optional }
        );

        //routes.MapRoute(
        //  name: "Controllers",
        //  url: "{controller}/{action}/{id}",
        //  defaults: new { controller = "Error", action = "Index", code = UrlParameter.Optional }
        //);

        routes.MapRoute(
            name: "Default",
            url: "{controller}/{action}/{id}",
            defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional }
        );

        //Keep at bottom
        routes.MapRoute("CatchAll", "{*url}", new { controller = "Error", action = "Index", name = "no-route-found", code = "404" });

    }
}
底部的

CatchAll很好地捕获了所有与前面的路线不匹配的东西。

我还有很多测试场景,但困扰我的是以下 UrlParameter:

http://localhost:64275/does/not/exist/

上面的URL本质上就是http://localhost:64275/{controller}/{action}/{id}

我没有名为 does 的控制器,我认为如果没有控制器,defaults: new { controller = "Home", action = "Index", id = UrlParameter.Optional } 将默认为 Home 控制器,操作为 Index匹配。

另一个有效的例子:

http://localhost:64275/a/a/a/a/ (because it has 4 parts, not 3 or less)

有人可以解释我可能哪里出错了吗? ...还有什么我不明白的?

我应该实施这样的事情吗:.Net MVC Routing Catchall not working(Darin Dimitrov 的回答)

protected void Application_Error(object sender, EventArgs e)
{
    Exception exception = Server.GetLastError();
    HttpException httpException = exception as HttpException;
    if (httpException != null)
    {
        RouteData routeData = new RouteData();
        routeData.Values.Add("controller", "Error");
        routeData.Values.Add("action", "HttpError500");

            if (httpException.GetHttpCode() == 404)
            {
                routeData.Values["action"] = "HttpError404";
            }

        Server.ClearError();
        Response.Clear();
        IController errorController = new ErrorController();
        errorController.Execute(new RequestContext(new HttpContextWrapper(Context), routeData));
    }
}

您是正确的,因为在 http://localhost:64275/does/not/exist/ 情况下命中了默认路由。但是,路由没有内置任何昂贵的反射调用来确保控制器在尝试创建之前存在。

但是,您可以通过自定义 IControllerFactory 进行干预,当 MVC 无法找到路由中指定的控制器实例时知道该怎么做。

NotFoundControllerFactory.cs

public class NotFoundControllerFactory : DefaultControllerFactory
{
    protected override IController GetControllerInstance(RequestContext requestContext, Type controllerType)
    {
        // If controller not found it will be null, so we want to take control
        // of the request and send it to the ErrorController.NotFound method.
        if (controllerType == null)
        {
            requestContext.RouteData.Values["action"] = "NotFound";
            requestContext.RouteData.Values["controller"] = "Error";
            return base.GetControllerInstance(requestContext, typeof(ErrorController));
        }

        return base.GetControllerInstance(requestContext, controllerType);
    }
}

DefaultControllerFactory 接受控制器作为字符串,然后尝试将控制器名称解析为类型。如果不能,则调用 GetControllerInstance(RequestContext, Type) 时类型为 null。因此,我们所要做的就是检查类型是否为 null,然后重写我们的请求,使其实例化 ErrorController 并调用 NotFound 操作方法。

用法

您只需要在启动时向 MVC 注册控制器工厂。

ControllerBuilder.Current.SetControllerFactory(new NotFoundControllerFactory());

然后将所有内容联系在一起,您的 NotFound 操作方法应将状态代码设置为 404。

public class ErrorController : Controller
{
    public ActionResult NotFound()
    {
        Response.StatusCode = (int)System.Net.HttpStatusCode.NotFound;
        Response.StatusDescription = "404 Not Found";

        return View();
    }
}

至于操作方法,如果控制器上不存在操作,MVC 以 404 Not Found 响应。