【C#】WinForm如何优雅的处理异常
|
admin
2023年8月1日 11:39
本文热度 531
|
场景:在运行、调试代码的时候,往往会遇到一些异常情况,很多时候我们代码写得足够健壮,所以出现的概率比较低,但是如果一旦出现,我们又没处理好,就可能会导致程序的崩溃、退出;当然还有一部分在方法内被我们用try{} catch{}
给处理掉了,但是不得不承认,还是会有极少的异常未被规避,或者说,相信不是所有的方法内都做了异常处理,所以如何处理这些异常就显得尤为重要
需求:能否提供一个捕捉全局异常的方式,并且最好能知道异常出在哪个位置?这样首先可以规避掉未处理的异常,增强程序的可用性,然后一旦出现问题又能够精准定位、方便排查。
开发环境:.NET Framework版本:4.5
开发工具: Visual Studio 2013
实现代码:
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
BindExceptionHandler();
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new Form1());
}
/// <summary>
/// 绑定程序中的异常处理
/// </summary>
private static void BindExceptionHandler()
{
//设置应用程序处理异常方式:ThreadException处理
Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
//处理UI线程异常
Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
//处理未捕获的异常
AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);
}
/// <summary>
/// 处理UI线程异常
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
{
string str = GetExceptionMsg(e.Exception, e.ToString());
MessageBox.Show(str, "系统错误", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
/// <summary>
/// 处理未捕获的异常
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
{
string str = GetExceptionMsg(e.ExceptionObject as Exception, e.ToString());
MessageBox.Show(str, "系统错误", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
/// <summary>h
/// 生成自定义异常消息
/// </summary>
/// <param name="ex">异常对象</param>
/// <param name="backStr">备用异常消息:当ex为null时有效</param>
/// <returns>异常字符串文本</returns>
private static string GetExceptionMsg(Exception ex, string backStr)
{
StringBuilder sb = new StringBuilder();
sb.AppendLine("****************************异常文本****************************");
sb.AppendLine("【出现时间】:" + DateTime.Now);
if (ex != null)
{
sb.AppendLine("【异常类型】:" + ex.GetType().Name);
sb.AppendLine("【异常信息】:" + ex.Message);
sb.AppendLine("【堆栈调用】:" + ex.StackTrace);
sb.AppendLine("【异常方法】:" + ex.TargetSite);
}
else
{
sb.AppendLine("【未处理异常】:" + backStr);
}
sb.AppendLine("***************************************************************");
return sb.ToString();
}
private void button1_Click(object sender, EventArgs e)
{
int x = int.Parse(textBox1.Text);
}
实现效果:
代码解析:首先就是在Program文件中新建一个函数:BindExceptionHandler
,在此方法绑定程序中的异常处理,即UI线程异常方法:Application_ThreadException
和未捕获的异常处理方法CurrentDomain_UnhandledException
。最后再在Main函数中调用即可。然后在Form中随便写个可造成异常的代码,即效果如上图。
另外在方法内部处理返回逻辑的时候,也可以考虑使用异常,既保证了函数的健壮性,又可以统一处理错误,即将所有的非正确的值以异常的方式抛出来,而不是return;代码如下:
private string GetMsg(int code)
{
try
{
if (code == 10)
{
return "Success";
}
else
{
throw new Exception("Error Code");
}
}
catch (Exception ex)
{
return "Error:" + ex.Message;
}
}
该文章在 2023/8/1 11:39:28 编辑过