您现在的位置是:亿华云 > 数据库

Linux+Windows: 程序崩溃时,在 C++ 代码中,如何获取函数调用栈信息

亿华云2025-10-04 00:43:01【数据库】1人已围观

简介一、前言程序在执行过程中 crash 是非常严重的问题,一般都应该在测试阶段排除掉这些问题,但是总会有漏网之鱼被带到 release 阶段。因此,程序的日志系统需要侦测这种情况,在代码崩溃的时候获取函

一、程序前言

程序在执行过程中 crash 是崩溃非常严重的问题,一般都应该在测试阶段排除掉这些问题,代码调用但是中何栈信总会有漏网之鱼被带到 release 阶段。

因此,获取函数程序的程序日志系统需要侦测这种情况,在代码崩溃的崩溃时候获取函数调用栈信息,为 debug 提供有效的网站模板代码调用信息。

这篇文章的中何栈信理论知识很少,直接分享 2 段代码:在 Linux 和 Windows 这 2 个平台上,获取函数如何用C++ 来捕获函数调用栈里的程序信息。

二、崩溃Linux 平台

1. 注册异常信号的代码调用处理函数

需要处理哪些异常信号

#include <execinfo.h> #include <cxxabi.h> #include <signal.h> const std::map<int, std::string> Signals = {      { SIGINT, "SIGINT"},         { SIGABRT, "SIGABRT"},      { SIGFPE, "SIGFPE"},        { SIGILL, "SIGILL"},       { SIGSEGV, "SIGSEGV"}     // 可以添加其他信号 }; 

注册信号处理函数

struct sigaction action; sigemptyset(&action.sa_mask); action.sa_sigaction = &sigHandler; action.sa_flags = SA_SIGINFO;   for (const auto &sigPair : Signals)  {      if (sigaction(sigPair.first, &action, NULL) < 0)         fprintf(stderr, "Error: sigaction failed! \n");  } 

2. 捕获异常,获取函数调用栈信息

void sigHandler(int signum,中何栈信 siginfo_t *info, void *ctx) {      const size_t dump_size = 50;     void *array[dump_size];     int size = backtrace(array, dump_size);     char **symbols = backtrace_symbols(array, size);     std::ostringstream oss;     for (int i = 0; i < size; ++i)     {          char *mangleName = 0;         char *offsetBegin = 0;         char *offsetEnd = 0;         for (char *p = symbols[i]; *p; ++p)         {              if (( == *p)             {                      mangleName = p;             }                else if (+ == *p)             {                  offsetBegin = p;             }             else if () == *p)             {                  offsetEnd = p;                 break;             }         }         if (mangleName && offsetBegin && offsetEnd && mangleName < offsetBegin)         {              *mangleName++ = \0;             *offsetBegin++ = \0;             *offsetEnd++ = \0;             int status;             char *realName = abi::__cxa_demangle(mangleName, 0, 0, &status);             if (0 == status)                 oss << "\tstack dump [" << i << "]  " << symbols[i] << " : " << realName << "+";             else                 oss << "\tstack dump [" << i << "]  " << symbols[i] << mangleName << "+";             oss << offsetBegin << offsetEnd << std::endl;             free(realName);         }         else         {              oss << "\tstack dump [" << i << "]  " << symbols[i] << std::endl;         }     }     free(symbols);     oss << std::endl;     std::cout << oss.str(); // 打印函数调用栈信息 } 

三、Windwos 平台

在 Windows 平台下的获取函数代码实现,高防服务器参考了国外某个老兄的代码,如下:

1. 设置异常处理函数

#include <windows.h> #include <dbghelp.h> SetUnhandledExceptionFilter(exceptionHandler); 

2. 捕获异常,获取函数调用栈信息

void exceptionHandler(LPEXCEPTION_POINTERS info) {      CONTEXT *context = info->ContextRecord;     std::shared_ptr<void> RaiiSysCleaner(nullptr, [&](void *) {        SymCleanup(GetCurrentProcess());     });   const size_t dumpSize = 64;   std::vector<uint64_t> frameVector(dumpSize);   DWORD machine_type = 0;   STACKFRAME64 frame = { };   frame.AddrPC.Mode = AddrModeFlat;   frame.AddrFrame.Mode = AddrModeFlat;   frame.AddrStack.Mode = AddrModeFlat; #ifdef _M_IX86   frame.AddrPC.Offset = context->Eip;   frame.AddrFrame.Offset = context->Ebp;   frame.AddrStack.Offset = context->Esp;   machine_type = IMAGE_FILE_MACHINE_I386; #elif _M_X64   frame.AddrPC.Offset = context->Rip;   frame.AddrFrame.Offset = context->Rbp;   frame.AddrStack.Offset = context->Rsp;   machine_type = IMAGE_FILE_MACHINE_AMD64; #elif _M_IA64   frame.AddrPC.Offset = context->StIIP;   frame.AddrFrame.Offset = context->IntSp;   frame.AddrStack.Offset = context->IntSp;   machine_type = IMAGE_FILE_MACHINE_IA64;   frame.AddrBStore.Offset = context.RsBSP;   frame.AddrBStore.Mode = AddrModeFlat; #else   frame.AddrPC.Offset = context->Eip;   frame.AddrFrame.Offset = context->Ebp;   frame.AddrStack.Offset = context->Esp;   machine_type = IMAGE_FILE_MACHINE_I386; #endif   for (size_t index = 0; index < frameVector.size(); ++index)   {      if (StackWalk64(machine_type,            GetCurrentProcess(),            GetCurrentThread(),            &frame,            context,            NULL,            SymFunctionTableAccess64,            SymGetModuleBase64,            NULL)) {        frameVector[index] = frame.AddrPC.Offset;     } else {        break;     }   }   std::string dump;   const size_t kSize = frameVector.size();   for (size_t index = 0; index < kSize && frameVector[index]; ++index) {      dump += getSymbolInfo(index, frameVector);     dump += "\n";   } std::cout << dump;  } 

主要是利用了 StackWalk64 这个函数,从地址转换为函数名称。

利用以上几个神器,基本上可以获取到程序崩溃时的函数调用栈信息,定位问题,有如神助!

很赞哦!(32)