【问题标题】:How can I handle errors in Global.asax Application_Start?如何处理 Global.asax Application_Start 中的错误?
【发布时间】:2023-03-26 11:07:02
【问题描述】:

为了在用户访问网站时始终向他们显示“友好”错误页面,我在 Global.asax 页面中提供了一个包罗万象的信息,大多数错误都由过滤器处理,这似乎是首选方法。在大多数情况下,这工作正常。但是,在 Application_Start 期间,Application_Error 事件(可以理解)不会被触发。

我的 Application_Start 事件包含一些依赖于服务调用的初始化代码,因此一个容易定义的故障点是如果服务由于某种原因不可用。我发现解决此问题的唯一方法是执行以下操作。

    private static Exception StartUpException;
    protected void Application_Start()
    {
        AreaRegistration.RegisterAllAreas();
        FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
        RouteConfig.RegisterRoutes(RouteTable.Routes);
        Initialise();       
    }

    private void Initialise()
    {
        StartUpException = null;
        try
        {
            Bootstrapper.Initialise();
        }
        catch (Exception ex)
        {
            StartUpException = ex;
        }
    }

然后我在 Application_BeginRequest 中有以下代码

    protected void Application_BeginRequest(object sender, EventArgs e)
    {
        if (StartUpException != null)
        {
            HandleErrorAndRedirect(StartUpException);
            HttpRuntime.UnloadAppDomain();
            Response.End();
        }
    }

这行得通,但似乎有点 hack。我也不确定调用 UnloadAppDomain 的后果,或者如果多个请求到达会发生什么。有没有更好的方法来管理这个?

【问题讨论】:

  • GlobalFilters.Filters 是否包含HandleError?如果是这样,那就是捕获所有错误。
  • 不,我已经删除了。
  • 请原谅我的说法,但是让您的 application_start 依赖于服务肯定有点 [恕我直言] 不好的举动?有没有办法把它移到别处?由于App启动有点重!我还希望如果它确实失败然后卸载,那么下一个进入的请求将要么等待 app_start 完成然后拾取线程,要么调用另一个 App_Start 然后失败..
  • 你可能是对的,但我不确定我还能把它放在哪里。在 Initialise 代码中,我正在配置 Unity 容器,它将为每个控制器注入服务的包装类。包装类本身包含几个依赖项,它们在其构造函数中从服务加载数据。这些依赖项被配置为单例,所以这个负载应该只发生一次,而不是每个请求。

标签: c# asp.net asp.net-mvc-4 error-handling global-asax


【解决方案1】:

我们在 App_Start 中的引导存在问题,因为 HttpContext 没有设置,并且一些引导类需要它;无论如何,这也适用于您的情况:

public class MvcApplication : System.Web.HttpApplication {    
    protected void Application_BeginRequest() {
        var context = this.Context;
        FirstTimeInitializer.Init(context);
    }

    private static class FirstTimeInitializer {
        private static bool s_IsInitialized = false;
        private static Object s_SyncRoot = new Object();

        public static void Init(HttpContext context) {
            if (s_IsInitialized) {
                return;
            }

            lock (s_SyncRoot) {
                if (s_IsInitialized) {
                    return;
                }

                // bootstrap

                s_IsInitialized = true;
            }
        }
    }
}

【讨论】:

    最近更新 更多