VC++中的异常处理堆栈跟踪

exception handling stack trace in VC++

本文关键字:堆栈 跟踪 异常处理 VC++      更新时间:2023-10-16

我是C++编程的新手_当我将无效参数传递给C++系统函数时,set_invalid_parameter_handler会给我函数名、文件和行,例如:在_tmain printf中,行生成错误,但我想获得编程的函数名、文档和行,而不是系统函数。

void myInvalidParameterHandler(const wchar_t* expression,
    const wchar_t* function,
    const wchar_t* file,
    unsigned int line,
    uintptr_t pReserved)
{
    wprintf(L"Invalid parameter detected in function %s."
        L" File: %s Line: %dn", function, file, line);
    wprintf(L"Expression: %sn", expression);
    call_stack st;
    st.stack.back();
    cout << st.stack.back().to_string();
    cout << st.to_string();
}
int _tmain(int argc, _TCHAR* argv[])
{
    char* formatString;
    _invalid_parameter_handler oldHandler, newHandler;
    newHandler = myInvalidParameterHandler;
    oldHandler = _set_invalid_parameter_handler(newHandler);
    formatString = NULL;
    printf(formatString); // I want to get this line
    return 0;
}

在此示例中,myInvalidParameterHandler生成以下输出:函数=printf,文件=f:\dd\vctools\crt\crtw32\stdio\printf.c,表达式=(format!=NULL),行=54

但我想得到这样的东西:函数=_tmain,文件:。。。\MySample.cpp,line=(MySample.cpp中的printf行)我该怎么做?(像C#堆栈跟踪)?

我还尝试了stackwalker(call_stack),它给了我自己的线路,我无法通过这种方式获得我的需求。

编辑:还没有回答任何人。请帮忙。2015.02.25 16:42土耳其(雅典地区)

问题是,在C++中,调用堆栈不像C#中那样容易使用。

要在向公众发布的应用程序中获得调用堆栈,您需要原始源代码、原始DLL文件、原始PDB文件和转储文件。

您可以使用崩溃处理程序中的MiniDumpWriteDump()在代码中生成转储文件。但这需要一个结构化的例外。

所以你必须写:

#define SEH_ARG_EXCEPTION 0xE0415247 // "ARG"
void myInvalidParameterHandler(....) // all params are NULL in a release build
{
    RaiseException(SEH_ARG_EXCEPTION, EXCEPTION_NONCONTINUABLE, 0, NULL);
}

在一个结构化的异常处理程序中,你会发现:

__try
{
    Execute your code with invalid argument 
}
__except(ExceptionHandler(GetExceptionInformation())) 
{
}
static int ExceptionHandler(EXCEPTION_POINTERS* pk_Ptr)
{
    switch (pk_Ptr->ExceptionRecord->ExceptionCode)
    {
        case EXCEPTION_BREAKPOINT:
        case EXCEPTION_SINGLE_STEP:
            return EXCEPTION_CONTINUE_SEARCH;
    }
    MINIDUMP_EXCEPTION_INFORMATION k_Mini; 
    k_Mini.ThreadId          = GetCurrentThreadId(); 
    k_Mini.ExceptionPointers = pk_Ptr; 
    k_Mini.ClientPointers    = FALSE; 
    HANDLE h_File = CreateFile(L"DumpPath\XYZ.dmp", GENERIC_READ | GENERIC_WRITE, 0, NULL, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, NULL)
    MiniDumpWriteDump(GetCurrentProcess(), GetCurrentProcessId(), h_File, MINIDUMP_TYPE, &k_Mini, 0, 0);
    CloseHandle(h_File);
    MessageBox(NULL, L"The application has crashed.nA Minidump has been written to: XYZ.dmpnPlease send this file to xyz@abc.com", L"Fatal Error", MB_ICONSTOP | MB_TOPMOST);
    TerminateProcess(GetCurrentProcess(), 0);
    return EXCEPTION_EXECUTE_HANDLER;
}

然后,您将在磁盘上有一个DMP文件,用户必须将该文件发送给您,以便将其加载到Visual Studio中,并从堆栈中获取崩溃位置。

重要提示:您需要确切的原始代码、DLL和PDB文件才能正常工作。