我试图在我的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/
以上网址基本上是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));
}
}
答案 0 :(得分:2)
你是正确的,因为在http://localhost:64275/does/not/exist/
案例中正在点击默认路由。但是,在尝试创建控制器之前,路由没有内置任何昂贵的Reflection调用以确保控制器存在。
但是,您可以通过创建自己的自定义IControllerFactory
进行干预,该自定义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);
}
}
知道当MVC无法找到路由中指定的控制器实例时该怎么做。
DefaultControllerFactory
null
接受控制器作为字符串,然后尝试将控制器名称解析为类型。如果在调用GetControllerInstance(RequestContext, Type)
时类型不是null
。因此,我们所要做的就是检查类型是否为ErrorController
,然后重写我们的请求,以便实例化NotFound
并调用ControllerBuilder.Current.SetControllerFactory(new NotFoundControllerFactory());
操作方法。
您只需要在启动时使用MVC注册控制器工厂。
NotFound
然后,要将所有内容绑定在一起,您的public class ErrorController : Controller
{
public ActionResult NotFound()
{
Response.StatusCode = (int)System.Net.HttpStatusCode.NotFound;
Response.StatusDescription = "404 Not Found";
return View();
}
}
操作方法应将状态代码设置为404。
extract
对于操作方法,如果控制器上不存在该操作,MVC 会以404 Not Found响应。