大部分具有 Exception 機制的程式語言都有提供全域的 Exception 處理,如果你已經有用 log 去記錄錯誤的習慣的話,與其在程式裡佈滿了 try catch,不如在全域處裡中去記錄沒有被 catch 的 Exception,這樣程式就可以更乾淨了,而且程式如果發生異常的關閉時也會進入全域處裡,可以確保 Exception 妥善地被記錄。
Console, WinForm 程式
using System; using System.Security.Permissions; using System.Threading; using System.Windows.Forms; using NLog; namespace AppThreadException { static class Program { private static readonly ILogger _log = LogManager.GetCurrentClassLogger(); [STAThread] [SecurityPermission(SecurityAction.Demand, Flags = SecurityPermissionFlag.ControlAppDomain)] public static void Main(string[] args) { /* ThreadException 用來攔截 UI 錯誤 */ Application.ThreadException += threadExceptionHandler; /* UnhandledException 只能攔截錯誤,不能阻止程式關閉 */ Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException); AppDomain.CurrentDomain.UnhandledException += unhandledExceptionHandler; Application.Run(new MyForm()); } /// <summary>攔截 UI 錯誤</summary> private static void threadExceptionHandler(object sender, ThreadExceptionEventArgs e) { _log.Fatal(e.Exception, "操作錯誤"); MessageBox.Show(e.Exception.Message, "操作錯誤", MessageBoxButtons.OK, MessageBoxIcon.Stop); } /// <summary>攔截不可挽回的錯誤,不能阻止程式關閉</summary> private static void unhandledExceptionHandler(object sender, UnhandledExceptionEventArgs e) { Exception ex = (Exception)e.ExceptionObject; _log.Fatal(ex, "執行錯誤"); MessageBox.Show(ex.Message, "執行錯誤", MessageBoxButtons.OK, MessageBoxIcon.Stop); } } }
Web 程式
在 Global.asax.cs 中可以設定 Application_Error 就可以攔截未處裡的 Exception,除了用這個方法記錄錯誤,還可以用現有的套件(elmah)幫我們完成。
using System; using System.Web.Mvc; using System.Web.Optimization; using System.Web.Routing; using NLog; namespace MvcReport { public class MvcApplication : System.Web.HttpApplication { private static readonly ILogger _log = LogManager.GetCurrentClassLogger(); //... protected void Application_Error(object sender, EventArgs e) { Exception ex = Server.GetLastError(); _log.Fatal(ex, ex.Message); } //... } }
Net core 程式
Net core 的程式都是相同的方式,Web Request 的要另外配置
using System; using System.IO; using System.Threading; using System.Threading.Tasks; using Autofac.Extensions.DependencyInjection; using EverTop.Api; using Microsoft.AspNetCore.Hosting; using Microsoft.Extensions.Hosting; using NLog; namespace MyWebApp { public class Program { private static readonly ILogger _log = LogManager.GetCurrentClassLogger(); public static void Main(string[] args) { /* UnhandledException 只能攔截錯誤,不能阻止程式關閉 */ AppDomain.CurrentDomain.UnhandledException += unhandledExceptionHandler; /* 用來攔截 Task 錯誤 */ TaskScheduler.UnobservedTaskException += unobservedTaskException; var host = Host.CreateDefaultBuilder(args) .UseServiceProviderFactory(new AutofacServiceProviderFactory()) .ConfigureWebHostDefaults(webHostBuilder => webHostBuilder .UseStartup<Startup>() ) .Build(); host.Run(); /* 啟動網站 */ } private static void unhandledExceptionHandler(object sender, UnhandledExceptionEventArgs e) { Exception ex = (Exception)e.ExceptionObject; _log.Fatal(ex, "執行錯誤"); } private static void unobservedTaskException(object sender, UnobservedTaskExceptionEventArgs e) { _log.Fatal(e.Exception, "執行錯誤"); e.SetObserved(); } } }
Net core 3.1 Web 程式
在 Startup.cs 中配置 middleware 進行 Exception 攔截
public void Configure(IApplicationBuilder app, IWebHostEnvironment env) { if (_env.IsProduction()) { app.UseExceptionHandler("/Error"); } else { app.UseDeveloperExceptionPage(); } app.UseStatusCodePagesWithReExecute("/Error"); /* 利用 middleware 進行 Exception 攔截 */ /* 這裡的順序很重要,不然會被前面 ExceptionHandler 處理掉就拿不到 Exception */ app.Use(async (context, next) => { try { await next(); } catch (Exception ex) { _log.Fatal(ex, "執行錯誤"); throw; /* 把 Exception 再丟出去給別人處理 */ } }); }
週期性 Thread 的處裡方式
將主要邏輯寫在另外 method 裡,這樣可以專注在 Exception 上。
private bool _runFlag = false; public void Start() { if (_runFlag) { return; } _runFlag = true; var thread = new Thread(() => { while (_runFlag) { try { cycleHandler(); } catch (Exception ex) { _log.Fatal(ex, "執行錯誤"); } Thread.Sleep(1000); } }); thread.Start(); } public void Stop() { _runFlag = false; } private void cycleHandler() { // 主要的邏輯程式寫在這裡 }
PHP 錯誤處理
Ref: set_error_handler, set_exception_handler
<?php function error_handler($errno, $errstr, $errfile, $errline) { if(error_reporting() === 0){ return; } throw new ErrorException($errstr, 0, $errno, $errfile, $errline); } set_error_handler('error_handler', E_ALL^E_NOTICE); function exception_handler($exception) { // 在這記錄 log throw $exception; } set_exception_handler('exception_handler'); throw new Exception('Uncaught Exception'); //$a = 1 / 0;
0 回應:
張貼留言