在调试的时候出现莫名的异常,但是不知道具体出错的位置,编译运行环境也不报错。可以在主窗体加载并初始化前添加日志打印异常信息代码。
代码移植,先把原先的加载代码删去,重点是在代码里面改成我们应用程序的入口就可以。日志打印异常信息代码方便排查代码中隐藏的bug,比如数组越界访问的问题。就拿数组越界来说,程序在运行,莫名的某个时间点,在某个方法中数组访问越界了,会弹出异常信息对话框,里面会提示出错的源文件位置及那个方法那个行导致的bug。可以让我们高效的解决代码隐含致命问题。
using System;
using System.Collections.Generic;
using System.Linq;
using System.Windows.Forms;
using System.Text;
namespace Demo //工程名
{
/// <summary>
/// 此类用于捕获Application异常信息
/// </summary>
class ApplicationException
{
/// <summary>
/// 定义委托接口处理函数,调用此类中的Main函数为应用添加异常信息捕获
/// </summary>
public delegate void ExceptionCall();
public static void Run(ExceptionCall exCall)
{
try
{
//设置应用程序处理异常方式:ThreadException处理
Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
//处理UI线程异常
Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
//处理非UI线程异常
AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);
if (exCall != null) exCall();
}
catch (Exception ex)
{
string str = GetExceptionMsg(ex, string.Empty);
MessageBox.Show(str, "系统错误", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
}
///--------------------------------------------------------------------------
/ <summary>
/ 应用程序的主入口点。
/ </summary>
//[STAThread]
//static void Main()
//{
// try
// {
// //设置应用程序处理异常方式:ThreadException处理
// Application.SetUnhandledExceptionMode(UnhandledExceptionMode.CatchException);
// //处理UI线程异常
// Application.ThreadException += new System.Threading.ThreadExceptionEventHandler(Application_ThreadException);
// //处理非UI线程异常
// AppDomain.CurrentDomain.UnhandledException += new UnhandledExceptionEventHandler(CurrentDomain_UnhandledException);
// #region 应用程序的主入口点
// Application.EnableVisualStyles();
// Application.SetCompatibleTextRenderingDefault(false);
// Application.Run(new Form1());
// #endregion
// }
// catch (Exception ex)
// {
// string str = GetExceptionMsg(ex, string.Empty);
// MessageBox.Show(str, "系统错误", MessageBoxButtons.OK, MessageBoxIcon.Error);
// }
//}
static void Application_ThreadException(object sender, System.Threading.ThreadExceptionEventArgs e)
{
string str = GetExceptionMsg(e.Exception, e.ToString());
MessageBox.Show(str, "系统错误", MessageBoxButtons.OK, MessageBoxIcon.Error);
}
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>
/// 生成自定义异常消息
/// </summary>
/// <param name="ex">异常对象</param>
/// <param name="backStr">备用异常消息:当ex为null时有效</param>
/// <returns>异常字符串文本</returns>
static string GetExceptionMsg(Exception ex, string backStr)
{
StringBuilder sb = new StringBuilder();
sb.AppendLine("****************************异常文本****************************");
sb.AppendLine("【出现时间】:" + DateTime.Now.ToString("yyyy-MM-dd hh:mm:ss"));
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();
}
}
static class Program
{
/// <summary>
/// 应用程序的主入口点。
/// </summary>
[STAThread]
static void Main()
{
ApplicationException.Run(call); // 调用异常信息捕获类,进行异常信息的捕获
}
// 应用程序,入口逻辑
public static void call()
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new myForm()); //应用程序的主入口点。
}
}
}