博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
借助backtrace和demangle实现异常类Exception
阅读量:6487 次
发布时间:2019-06-24

本文共 4934 字,大约阅读时间需要 16 分钟。

C++的异常类是没有栈痕迹的,如果需要获取栈痕迹,需要使用以下函数:

#include 
int backtrace(void **buffer, int size);char **backtrace_symbols(void *const *buffer, int size);void backtrace_symbols_fd(void *const *buffer, int size, int fd);

backtrace将当前程序的调用信息存储在buffer中,backtrace_symbols则是将buffer翻译为字符串。后者用到了malloc,所以需要手工释放内存。

man手册中提供了如下的代码:

#include 
#include
#include
#include
voidmyfunc3(void){ int j, nptrs;#define SIZE 100 void *buffer[100]; char **strings; nptrs = backtrace(buffer, SIZE); printf("backtrace() returned %d addresses\n", nptrs); /* The call backtrace_symbols_fd(buffer, nptrs, STDOUT_FILENO) would produce similar output to the following: */ strings = backtrace_symbols(buffer, nptrs); if (strings == NULL) { perror("backtrace_symbols"); exit(EXIT_FAILURE); } for (j = 0; j < nptrs; j++) printf("%s\n", strings[j]); free(strings);}static void /* "static" means don't export the symbol... */myfunc2(void){ myfunc3();}voidmyfunc(int ncalls){ if (ncalls > 1) myfunc(ncalls - 1); else myfunc2();}intmain(int argc, char *argv[]){ if (argc != 2) { fprintf(stderr, "%s num-calls\n", argv[0]); exit(EXIT_FAILURE); } myfunc(atoi(argv[1])); exit(EXIT_SUCCESS);}

编译并执行:

$  cc -rdynamic prog.c -o prog$ ./prog 3

输出如下:

backtrace() returned 8 addresses./prog(myfunc3+0x1f) [0x8048783]./prog() [0x8048810]./prog(myfunc+0x21) [0x8048833]./prog(myfunc+0x1a) [0x804882c]./prog(myfunc+0x1a) [0x804882c]./prog(main+0x52) [0x8048887]/lib/i386-linux-gnu/libc.so.6(__libc_start_main+0xf3) [0xb76174d3]./prog() [0x80486d1]

 

因此我写出以下的异常类,注意上面的打印结果经过了名字改编,所以我们使用abi::__cxa_demangle将名字还原出来。

Exception.h

#ifndef EXCEPTION_H_#define EXCEPTION_H_#include 
#include
class Exception : public std::exception{public: explicit Exception(const char* what); explicit Exception(const std::string& what); virtual ~Exception() throw(); virtual const char* what() const throw(); const char* stackTrace() const throw();private: void fillStackTrace(); //填充栈痕迹 std::string demangle(const char* symbol); //反名字改编 std::string message_; //异常信息 std::string stack_; //栈trace};#endif // EXCEPTION_H_

Exception.cpp

#include "Exception.h"#include 
#include
#include
#include
using namespace std;Exception::Exception(const char* msg) : message_(msg){ fillStackTrace();}Exception::Exception(const string& msg) : message_(msg){ fillStackTrace();}Exception::~Exception() throw (){}const char* Exception::what() const throw(){ return message_.c_str();}const char* Exception::stackTrace() const throw(){ return stack_.c_str();}//填充栈痕迹void Exception::fillStackTrace(){ const int len = 200; void* buffer[len]; int nptrs = ::backtrace(buffer, len); //列出当前函数调用关系 //将从backtrace函数获取的信息转化为一个字符串数组 char** strings = ::backtrace_symbols(buffer, nptrs); if (strings) { for (int i = 0; i < nptrs; ++i) { // TODO demangle funcion name with abi::__cxa_demangle //strings[i]代表某一层的调用痕迹 stack_.append(demangle(strings[i])); stack_.push_back('\n'); } free(strings); }}//反名字改编string Exception::demangle(const char* symbol){ size_t size; int status; char temp[128]; char* demangled; //first, try to demangle a c++ name if (1 == sscanf(symbol, "%*[^(]%*[^_]%127[^)+]", temp)) { if (NULL != (demangled = abi::__cxa_demangle(temp, NULL, &size, &status))) { string result(demangled); free(demangled); return result; } } //if that didn't work, try to get a regular c symbol if (1 == sscanf(symbol, "%127s", temp)) { return temp; } //if all else fails, just return the symbol return symbol;}

测试代码如下:

#include "Exception.h"#include 
using namespace std;class Bar{ public: void test() { throw Exception("oops"); }};void foo(){ Bar b; b.test();}int main(){ try { foo(); } catch (const Exception& ex) { printf("reason: %s\n", ex.what()); printf("stack trace: %s\n", ex.stackTrace()); }}

打印结果如下:

reason: oopsstack trace: Exception::fillStackTrace()Exception::Exception(char const*)Bar::test()foo()./a.out(main+0xf)/lib/i386-linux-gnu/libc.so.6(__libc_start_main+0xf3)./a.out()

注意编译的时候,加上-rdynamic选项

 

有了这个类,我们可以在程序中这样处理异常:

try    {        //    }    catch (const Exception& ex)    {        fprintf(stderr, "reason: %s\n", ex.what());        fprintf(stderr, "stack trace: %s\n", ex.stackTrace());        abort();    }    catch (const std::exception& ex)    {        fprintf(stderr, "reason: %s\n", ex.what());        abort();    }    catch (...)    {        fprintf(stderr, "unknown exception caught \n");    throw; // rethrow    }

转载于:https://www.cnblogs.com/inevermore/p/4005489.html

你可能感兴趣的文章
ES6(正则扩展)
查看>>
从零玩转jQuery-核心函数和静态方法
查看>>
8月3日科技联播:新型VR头盔可令盲人重获光明 ,联通阿里成立合资企业“云粒智慧”...
查看>>
Jeknins的Pipline方式构建任务
查看>>
Python全栈 项目(电子词典、协程、pdb调试)
查看>>
(周期计划-7)常用集合的源码分析:ArrayList
查看>>
“分叉并商品化”,GitLab 和 Elastic 炮轰 AWS 的开源方法
查看>>
第162天:canvas中Konva库的使用方法
查看>>
设计模式学习(一):多用组合少用继承(C#)
查看>>
mysql 终端命令
查看>>
MKVToolNix v33.0.0 发布,MKV 视频编辑工具软件
查看>>
Egg 2.19.0 发布,阿里开源的企业级 Node.js 框架
查看>>
CentOS7.4安装Gitlab10.5.1及汉化
查看>>
Virtualbox以及VWare在Win10下的不兼容
查看>>
Github上如何在Fork到的开源项目中提交Pull requests?
查看>>
科箭TMS云,助力网易考拉跨境电商物流云平台
查看>>
Android ViewDragHelper:控制子View能否拖曳及水平方向的拖曳边界(2)
查看>>
Java单向链表反转
查看>>
java基础学习_反射、装饰模式、JDK新特性_day27总结
查看>>
Windows 10 又在生产环境进行测试?微软:发错了
查看>>