1、必备基础知识:三个与打印调用栈相关的函数
-
1.1、int backtrace(void** buffer, int size);
函数作用:用于获取当前线程的调用堆栈。
参数解释:
buffer:它是一个指针数组,函数获取的当前线程的调用堆栈将会被存放在buffer中。在buffer中的指针实际是从堆栈中获取的返回地址,每一个堆栈 框架有一个返回地址。
size:用来指定buffer中可以保存多少个void*元素。
函数返回值:实际获取的指针个数,最大不超过size大小。 -
1.2、char** backtrace_symbols (void *const *buffer, int size);
函数作用:将从backtrace函数获取的信息转化为一个字符串数组。
参数解释:
buffer:从backtrace函数获取的数组指针。
size:是该数组中的元素个数(backtrace函数的返回值)。
函数返回值:是一个指向字符串数组的指针,它的大小同buffer相同。每个字符串包含了一个相对于buffer中对应元素的 可打印信息。它包括函数名,函数的偏移地址,和实际的返回地址。
注:
1、只有使用ELF二进制格式的程序才能获取函数名称和偏移地址。在其他系统,只有16进制的返回地址能被获取。另外,需要传递相应的标志给链接器,以能支持函数名功能即编译选项-rdynamic。
2、backtrace_symbols生成的字符串都是malloc出来的,最后需要free该块内存。 -
1.3、void backtrace_symbols_fd (void *const *buffer, int size, int fd)
函数作用: backtrace_symbols_fd与backtrace_symbols函数具有相同的功能,不同的是它不会给调用者返回字符串数组,而是将结果写入文件描述符为fd的文件中,每个函数对应一行.它不需要调用malloc函数,因此适用于有可能调用该函数会失败的情况。
2、代码例子
#include <iostream>
#include <execinfo.h>
using namespace std;
void printCallStack()
{
int size = 16; // 最多打印16层调用栈
void* buffer[size] = {nullptr}; // 用于存放调用栈信息
int count = backtrace(buffer, size); // 获取调用栈
char** ptr = backtrace_symbols(buffer, count); // 将调用栈信息转成字符串
for (int i = 0; i< count; i++) { // 打印字符串
cout << ptr[i] << endl;
}
free(ptr); // 释放分配的内存
}
void test3(int n) { printCallStack(); }
void test2(int n) { test3(3); }
void test1(int n) { test2(2); }
int main() { test1(1); }
g++ -g -rdynamic bt_test.cpp
注意:
- 必须加编译选项-rdynamic,不然打印出的调用栈只有地址没有修饰后的函数名。
- 以上3个函数所在的头文件
#include <execinfo.h>
,只存在Linux系统下,Windows下用的头文件没去了解。 - backtrace_symbols通过malloc分配的内存,在使用完后需要free。
- 以上三个函数所在的头文件This header is Linux specific.
参考资料:
1、linux c 程序异常退出时打印堆栈调用信息
2、在C/C++程序中打印当前函数调用栈
3、stack overflow 问题:The execinfo.h header file does not exist