每当应用程序崩溃时,我都会尝试为它创建一个转储文件。我目前正在使用带有-e标志的procdump.exe,所以如果我的应用程序中有未处理的异常,procdump会为我创建一个转储文件。
我以为我已经完成了,但后来我发现我的应用程序崩溃了,procdump没有创建转储文件。经过一些调查,我发现vector::front的无效使用会导致运行时错误。我打开了_SECURE_SCL_THROWS标志,之后procdump.exe-e确实捕捉到了崩溃并创建了一个转储文件。
现在我的问题是:现在procdump.exe-e总是在我的应用程序崩溃时创建一个转储文件吗?我如何保证我没有任何其他情况下procdump-e对我不好?
我假设你在windows环境中(因为你使用procdump.exe)。你也可以为你的程序设置一个异常过滤器,每当你的应用程序崩溃时,它就会写一个mindump:
-
使用SetUnhandledExceptionFilter注册回调函数,该函数将在崩溃时调用。一个可能的签名是:
LONG WINAPI HandleException(struct _EXCEPTION_POINTERS* apExceptionInfo)
在某个地方注册使用:
SetUnhandledExceptionFilter(HandleException);
-
定义一个函数指针来调用函数MiniDumpWriteDump:
typedef BOOL (WINAPI *MINIDUMPWRITEDUMP)(HANDLE hProcess, DWORD dwPid, HANDLE hFile, MINIDUMP_TYPE DumpType,CONST PMINIDUMP_EXCEPTION_INFORMATION ExceptionParam,CONSTPMINIDUMP_USER_STREAM_INFORMATIOUserStreamParam,CONST PMINIDUMP_CALLBACK_INFORMATION CallbackParam);
-
使用MiniDumpWriteDump函数在以前注册的回调方法(HandleException)中写入转储(需要DbgHelp.dll 5.1或更高版本):
HMODULE mhLib = ::LoadLibrary(_T("dbghelp.dll")); MINIDUMPWRITEDUMP pDump = (MINIDUMPWRITEDUMP)::GetProcAddress(mhLib, "MiniDumpWriteDump"); HANDLE hFile = ::CreateFile(_T("dump_name"), GENERIC_WRITE, FILE_SHARE_WRITE, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL); _MINIDUMP_EXCEPTION_INFORMATION ExInfo; ExInfo.ThreadId = ::GetCurrentThreadId(); ExInfo.ExceptionPointers = apExceptionInfo; ExInfo.ClientPointers = FALSE; pDump(GetCurrentProcess(), GetCurrentProcessId(), hFile, MiniDumpNormal, &ExInfo, NULL, NULL); ::CloseHandle(hFile);
/*WinDump.cpp*/
#ifdef WIN32
#include <windows.h>
#include <Dbghelp.h>
#include <tchar.h>
typedef BOOL (WINAPI *MINIDUMPWRITEDUMP)(HANDLE hProcess, DWORD dwPid, HANDLE hFile, MINIDUMP_TYPE DumpType,CONST PMINIDUMP_EXCEPTION_INFORMATION ExceptionParam,CONST PMINIDUMP_USER_STREAM_INFORMATION UserStreamParam,CONST PMINIDUMP_CALLBACK_INFORMATION CallbackParam);
void create_minidump(struct _EXCEPTION_POINTERS* apExceptionInfo)
{
HMODULE mhLib = ::LoadLibrary(_T("dbghelp.dll"));
MINIDUMPWRITEDUMP pDump = (MINIDUMPWRITEDUMP)::GetProcAddress(mhLib, "MiniDumpWriteDump");
HANDLE hFile = ::CreateFile(_T("core.dmp"), GENERIC_WRITE, FILE_SHARE_WRITE, NULL, CREATE_ALWAYS,
FILE_ATTRIBUTE_NORMAL, NULL);
_MINIDUMP_EXCEPTION_INFORMATION ExInfo;
ExInfo.ThreadId = ::GetCurrentThreadId();
ExInfo.ExceptionPointers = apExceptionInfo;
ExInfo.ClientPointers = FALSE;
pDump(GetCurrentProcess(), GetCurrentProcessId(), hFile, MiniDumpNormal, &ExInfo, NULL, NULL);
::CloseHandle(hFile);
}
LONG WINAPI unhandled_handler(struct _EXCEPTION_POINTERS* apExceptionInfo)
{
create_minidump(apExceptionInfo);
return EXCEPTION_CONTINUE_SEARCH;
}
#endif // WIN32
/*WinDump.h*/
#ifdef WIN32
LONG WINAPI unhandled_handler(struct _EXCEPTION_POINTERS* apExceptionInfo);
#endif // WIN32
/*main.cpp*/
#include "WinDump.h"
int main(int argc, char **argv)
{
// Create a dump file whenever the gateway crashes only on windows
SetUnhandledExceptionFilter(unhandled_handler);
return 0;
}
或者使用Windows内置的错误报告WER,通过向注册表添加一个键,为您的应用程序启用它。更多详细信息请点击此处:https://learn.microsoft.com/en-us/windows/win32/wer/collecting-user-mode-dumps