打印当前函数的调用栈

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    

注意:

  1. 必须加编译选项-rdynamic,不然打印出的调用栈只有地址没有修饰后的函数名。
  2. 以上3个函数所在的头文件#include <execinfo.h>,只存在Linux系统下,Windows下用的头文件没去了解。
  3. backtrace_symbols通过malloc分配的内存,在使用完后需要free。
  4. 以上三个函数所在的头文件This header is Linux specific.

参考资料:
1、linux c 程序异常退出时打印堆栈调用信息
2、在C/C++程序中打印当前函数调用栈
3、stack overflow 问题:The execinfo.h header file does not exist

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值