C#WinForm程序異常退出的捕獲、繼續執行與自動重啓

本文參考網上搜索的信息,並作了適當修改可讓捕捉到異常以後阻止程序退出。html

另給出了經過命令行自動重啓的方法。spring

若是一個線程裏運行除以零的計算,以下面的代碼ide

        private void button1_Click(object sender, EventArgs e)
        {
            System.Threading.Thread t = new System.Threading.Thread(() =>
            {
                int a = 0;
                int c = 10 / a;
            });
            t.Start();
        }

 

將會致使程序自動結束,並且沒有任何提示信息 可是若是是在主線程裏運行這個代碼,是會彈出異常信息對話框的post

請問如何在線程裏也出現這個異常信息對話框.或者 避免程序直接退出,忽略異常,繼續往下執行呢?
 
在WINFORM主線程捕獲所有異常就行,以下代碼:
            //處理未捕獲的異常
            Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
            //處理UI線程異常
            Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
            //處理非UI線程異常
            AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);
 最常出現的錯誤在 :UnhandledException 裏出現。詳細代碼以下:
複製代碼
        /// <summary>
        /// 應用程序的主入口點。
        /// </summary>
        [STAThread]
        static void Main(string[] args) 
        {
            Application.EnableVisualStyles();
            Application.SetCompatibleTextRenderingDefault(false);

            //處理未捕獲的異常
            Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
            //處理UI線程異常
            Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
            //處理非UI線程異常
            AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);

            Application.Run(new Form1(args));
            glExitApp = true;//標誌應用程序能夠退出
        }

        /// <summary>
        /// 是否退出應用程序
        /// </summary>
        static bool glExitApp = false;

        static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
        {
            LogHelper.Save("CurrentDomain_UnhandledException", LogType.Error);
            LogHelper.Save("IsTerminating : " + e.IsTerminating.ToString(), LogType.Error);
            LogHelper.Save(e.ExceptionObject.ToString());

            while (true)
            {//循環處理,不然應用程序將會退出
                if (glExitApp) {//標誌應用程序能夠退出,不然程序退出後,進程仍然在運行
                    LogHelper.Save("ExitApp");
                    return; 
                }
                System.Threading.Thread.Sleep(2*1000);
            };
        }
        
        static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
        {
            LogHelper.Save("Application_ThreadException:" +
                e.Exception.Message, LogType.Error);
            LogHelper.Save(e.Exception);
            //throw new NotImplementedException();
        }
複製代碼
我本身稍微修改了點邏輯,你都懂的。
    static class Program
    {
        private static log4net.ILog _log = log4net.LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
        /// <summary>
        /// 是否退出應用程序
        /// </summary>
        static bool glExitApp = false;


        /// <summary>
        /// 應用程序的主入口點。
        /// </summary>
        [STAThread]
        static void Main()
        {
            _log.Info("程序啓動");

            try
            {
                //處理未捕獲的異常
                System.Windows.Forms.Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
                System.Windows.Forms.Application.ThreadException += Application_ThreadException;
                AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;

                Application.EnableVisualStyles();
                Application.SetCompatibleTextRenderingDefault(false);

                Application.Run(new WebManager());
                glExitApp = true;
            }
            catch (Exception ex)
            {
                _log.Error(ex);
            }
            _log.Info("程序關閉");

        }


        private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
        {
            _log.Error("CurrentDomain_UnhandledException ==== IsTerminating : " + e.IsTerminating.ToString());
            _log.Error("e.ExceptionObject : " + e.ExceptionObject);
            _log.Error(e);
            if (e.IsTerminating)
            {
                Common.Globalparams.ShowMessageDialog("系統發生錯誤,請聯繫系統管理員,程序即將關閉。");
                //循環處理,不然應用程序將會退出
                if (glExitApp)
                {
                    //標誌應用程序能夠退出,不然程序退出後,進程仍然在運行
                    _log.Error("====ExitApp");
                    return;
                }
                System.Threading.Thread.Sleep(2 * 1000);
                _log.Error("====UnhandledException  While......");

                Common.StartupHelper.CmdStartSelf();
            }
        }

        private static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
        {
            var ex = e.Exception;
            if (ex != null)
            {
                _log.Error(ex);
            }
        }

    }
View Code

 若是程序須要重啓只須要在捕獲的事件處理時啓動當前應用程序的代碼便可。參考以下:this

複製代碼
CmdStartCTIProc(Application.ExecutablePath, "cmd params");//放到捕獲事件的處理代碼後,重啓程序,須要時加上重啓的參數

        /// <summary>
        /// 在命令行窗口中執行
        /// </summary>
        /// <param name="sExePath"></param>
        /// <param name="sArguments"></param>
        static void CmdStartCTIProc(string sExePath, string sArguments)
        {
            Process p = new Process();
            p.StartInfo.FileName = "cmd.exe";
            p.StartInfo.UseShellExecute = false;
            p.StartInfo.RedirectStandardInput = true;
            p.StartInfo.RedirectStandardOutput = true;
            p.StartInfo.RedirectStandardError = true;
            p.StartInfo.CreateNoWindow = false;
            p.StartInfo.WindowStyle = System.Diagnostics.ProcessWindowStyle.Hidden;
            p.Start();
            p.StandardInput.WriteLine(sExePath + " " + sArguments);
            p.StandardInput.WriteLine("exit");
            p.Close();

            System.Threading.Thread.Sleep(2000);//必須等待,不然重啓的程序還未啓動完成;根據狀況調整等待時間
        }
複製代碼

 另一種重啓進程的方式:spa

            //重啓程序,須要時加上重啓的參數
            System.Diagnostics.ProcessStartInfo cp = new System.Diagnostics.ProcessStartInfo();
            cp.FileName = Application.ExecutablePath;
            cp.Arguments = "cmd params";
            cp.UseShellExecute = true;
            System.Diagnostics.Process.Start(cp);

完整代碼:命令行

    public class StartupHelper
    {
        /// <summary>
        /// 在命令行窗口中執行指定程序。(能夠包括程序自己)
        /// </summary>
        /// <param name="exePath"></param>
        /// <param name="sArguments"></param>
        public static void CmdStartApp(string exePath, string sArguments)
        {
            System.Diagnostics.Process p = new System.Diagnostics.Process();
            p.StartInfo.FileName = "cmd.exe";
            p.StartInfo.UseShellExecute = false;
            p.StartInfo.RedirectStandardInput = true;
            p.StartInfo.RedirectStandardOutput = true;
            p.StartInfo.RedirectStandardError = true;
            p.StartInfo.CreateNoWindow = false;
            p.StartInfo.WindowStyle = System.Diagnostics.ProcessWindowStyle.Hidden;
            p.Start();
            p.StandardInput.WriteLine(exePath + " " + sArguments);
            p.StandardInput.WriteLine("exit");
            p.Close();

            System.Threading.Thread.Sleep(2000);//必須等待,不然重啓的程序還未啓動完成;根據狀況調整等待時間
        }

        /// <summary>
        /// 啓動程序本身(通常程序異常退出時調用)
        /// </summary>
        /// <param name="strArguments"></param>
        public static void CmdStartSelf(string strArguments = "")
        {
            //重啓程序,須要時加上重啓的參數
            System.Diagnostics.ProcessStartInfo cp = new System.Diagnostics.ProcessStartInfo();
            cp.FileName = System.Windows.Forms.Application.ExecutablePath;
            cp.Arguments = strArguments;
            cp.UseShellExecute = true;
            System.Diagnostics.Process.Start(cp);

 } }
View Code

 

看了以爲有用的朋友,若是您方便的話,能夠頂一下。謝謝!線程

 

 

出處:https://www.cnblogs.com/zaspring/archive/2013/04/16/3023927.htmlcode

相關文章
相關標籤/搜索