C++记录程序崩溃时的dumpfile

 最近一段时间,新上线的软件在外场偶尔会出现异常崩溃的情况。由于试用范围比较分散,很难一一前往现场定位问题。而传统的log日志方法,在崩溃的情况下,并不能比较准确的表示出问题位置,这使得软件调试进程缓慢。

   后在公司前辈的指点下,我们想到了使用window自带的dumpfile来记录崩溃时刻的堆栈信息,这样配合log日志记录,能够快速的定位出问题点。大大提高了系统调试效率。

   经过一段时间的调试,现在项目已相对稳定了。想记录下此方法,以待后续类似情况下使用。

 
 
  1. //使所有版本都可以捕获到异常 
  2. void DisableSetUnhandledExceptionFilter() 
  3.     void *addr = (void*)GetProcAddress(LoadLibrary(_T("kernel32.dll")), "SetUnhandledExceptionFilter"); 
  4.  
  5.     if (addr)  
  6.     { 
  7.         unsigned char code[16]; 
  8.         int size = 0; 
  9.         code[size++] = 0x33; 
  10.         code[size++] = 0xC0; 
  11.         code[size++] = 0xC2; 
  12.         code[size++] = 0x04; 
  13.         code[size++] = 0x00; 
  14.  
  15.         DWORD dwOldFlag, dwTempFlag; 
  16.         VirtualProtect(addr, size, PAGE_READWRITE, &dwOldFlag); 
  17.         WriteProcessMemory(GetCurrentProcess(), addr, code, size, NULL); 
  18.         VirtualProtect(addr, size, dwOldFlag, &dwTempFlag); 
  19.     } 
  20.  
  21. //程序未捕获的异常处理函数 
  22. LONG WINAPI ExceptionFilter(struct _EXCEPTION_POINTERS *ExceptionInfo) 
  23.     ::AfxMessageBox("ExceptionFilter"); 
  24.  
  25.     HANDLE hFile = ::CreateFile( _T("C:\\dumpfile.dmp"), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); 
  26.     if( hFile != INVALID_HANDLE_VALUE) 
  27.     { 
  28.         MINIDUMP_EXCEPTION_INFORMATION einfo; 
  29.         einfo.ThreadId = ::GetCurrentThreadId(); 
  30.         einfo.ExceptionPointers = ExceptionInfo; 
  31.         einfo.ClientPointers = FALSE; 
  32.  
  33.         ::MiniDumpWriteDump(::GetCurrentProcess(), ::GetCurrentProcessId(), hFile, MiniDumpWithFullMemory, &einfo, NULL, NULL); 
  34.         ::CloseHandle(hFile); 
  35.     } 
  36.  
  37.     return 0; 
  38.  
  39. //把当前时刻的线程栈记录到DUMP文件中 
  40. int RecordCurStack() 
  41.     HANDLE hFile = ::CreateFile( _T("C:\\dumpfile.dmp"), GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); 
  42.     if( hFile != INVALID_HANDLE_VALUE) 
  43.     { 
  44.         ::MiniDumpWriteDump(::GetCurrentProcess(), ::GetCurrentProcessId(), hFile, MiniDumpWithFullMemory  ,NULL, NULL, NULL); 
  45.  
  46.         ::CloseHandle(hFile); 
  47.         return 1; 
  48.     } 
  49.  
  50.     return 0; 
  51.  
  52.  
  53. bool bCreateDumpThrd = true
  54. //循环检测线程 
  55. //查看到有ADTV2_TEMP.TXT文件,则记录下当前时刻的堆栈 
  56. void CreateDumpThrd(void* pv) 
  57.     HANDLE hFile;  
  58.     string strPath = FileAssist::GetExePath() + "\\ADTV2_TEMP.TXT"
  59.     while(bCreateDumpThrd) 
  60.     { 
  61.         //每5秒检测一次 
  62.         Sleep(5000); 
  63.         hFile = CreateFileA(strPath.c_str(),    // file to open 
  64.             GENERIC_READ,          // open for reading 
  65.             FILE_SHARE_READ,       // share for reading 
  66.             NULL,                  // default security 
  67.             OPEN_EXISTING,         // existing file only 
  68.             FILE_ATTRIBUTE_NORMAL, // normal file 
  69.             NULL);                 // no attr. template 
  70.  
  71.         if (hFile != INVALID_HANDLE_VALUE)  
  72.         {  
  73.             //防止多次记录当前堆栈信息,删除文件 
  74.             ::CloseHandle(hFile); 
  75.             ::DeleteFile(strPath.c_str()); 
  76.             RecordCurStack(); 
  77.         } 
  78.     } 

然后在程序入口将异常处理接口声明即可。

 
 
  1. //调试信息 
  2. ::SetUnhandledExceptionFilter(ExceptionFilter); //设置异常处理函数 
  3. DisableSetUnhandledExceptionFilter();           //获取未处理的异常 

这样,在程序异常时,就可以在C盘根目录下记录一个dumpfile.dmp的文件。这个文件会比较大,一般有100多M,其中信息比log形式的日志丰富很多,包括了异常时的堆栈调用关系以及各对象的值。,在VS中可以直接打开。如果保留了和当时编译软件一致的代码备份的话,可以直接使用VS的debug功能定位到问题代码行,否则,debug定位是到汇编代码行,看起来比较麻烦。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Windows程序崩溃,操作系统会生成一个崩溃转储(Dump)文件,用于帮助开发人员诊断和调试程序故障。这个Dump文件记录程序崩溃的内存状态,包括堆栈信息、寄存器状态、变量值等关键数据。 生成Dump文件的方法有多种,例如: 1. 使用Windows上自带的任务管理器。打开任务管理器,在“进程”选项卡中找到崩溃程序进程,右键点击选择“创建转储文件”即可生成Dump文件。 2. 使用Windows上自带的Dr.Watson工具(仅适用于旧版本)。Dr.Watson是一种活动监视工具,它会在程序崩溃自动记录信息,生成.DMP文件。可以在Windows注册表中启用Dr.Watson功能。 3. 使用Windows Debugging Tools。这是一套由微软提供的调试工具,其中包括了生成Dump文件的命令行工具Dumpchk、Msdia.dll等。使用这些工具可以对Dump文件进行详细的调试和分析。 一旦生成了Dump文件,开发人员可以使用各种调试工具来分析这个文件,以寻找程序崩溃的原因。比如,可以使用Visual Studio自带的调试器或WinDbg调试器来打开Dump文件,并逐步调试程序,查看导致崩溃的代码行。 Dump文件对于故障排除非常有用,可以帮助开发人员定位和解决程序中的错误。通过分析Dump文件,我们可以获得崩溃的内存状态,从而找到导致崩溃的具体代码,修复问题,提升软件的稳定性和可靠性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值