C++實(shí)現(xiàn)并優(yōu)化異常系統(tǒng)
C++原本的異常系統(tǒng)是這個(gè)樣子的:
調(diào)用what()方法時(shí)只返回異常的名稱,并沒有顯示拋出異常的位置和堆棧跟蹤,功能上顯得少許的貧瘠...
下面這個(gè)是我自己實(shí)現(xiàn)的改良版的異常處理系統(tǒng):
可以看到詳細(xì)的信息,下面是實(shí)現(xiàn)過(guò)程。
一、模擬棧展開的過(guò)程
網(wǎng)上看到別人用一些很奇怪的方法來(lái)獲取堆棧信息,從而實(shí)現(xiàn)堆棧跟蹤。
個(gè)人覺得很費(fèi)勁,而且還要安裝第三方庫(kù)。
于是我們可以寫一個(gè)類來(lái)模擬這個(gè)過(guò)程。
定義一個(gè)叫做ExceptionStackTrace的類:
class ExceptionStackTrace { private: const char** m_message_trace = nullptr; // 方法名數(shù)組 size_t* m_line_trace = nullptr; // 行數(shù)數(shù)組 int m_top; // 棧頂 int m_size; // 大小 public: ExceptionStackTrace(int size); void push(const char* message); // 入棧一個(gè)方法 void pop(); // 出棧一個(gè)方法 bool empty() const; // 判斷是否為空 int top() const; // 返回棧頂索引 int size() const; // 返回大小 void print_stack_trace() const; // 打印堆棧跟蹤信息 void throw_(SuperException except); // 拋出一個(gè)異常,需要繼承SuperException這個(gè)后面會(huì)講到 };
既然是模擬,所以需要在程序最前面定義一個(gè)靜態(tài)的對(duì)象,使用時(shí)在每一個(gè)函數(shù)的開始和結(jié)束部分加上這兩句:
static ExceptionStackTrace est = 128; void method(...) { est.push(__FUNCSIG__); ... // 異常拋出在這里可以被捕捉 est.pop(); } main ...
當(dāng)調(diào)用方法時(shí),會(huì)在調(diào)用ExceptionStackTrace的push方法,將方法信息壓棧。
之后再執(zhí)行方法內(nèi)部的語(yǔ)句。
最后在將方法出棧,模擬方法已被調(diào)用完畢。
下面是實(shí)現(xiàn)代碼:
ExceptionStackTrace::ExceptionStackTrace(int size) { // 尺寸不能是負(fù)數(shù) if (size <= 0) throw std::exception("Size should greater than 0."); m_message_trace = new const char*[size]; m_top = 0; m_size = size; } void ExceptionStackTrace::push(const char* message) { // 方法信息壓棧 m_message_trace[m_top] = message; ++m_top; } void ExceptionStackTrace::pop() { // 方法信息出棧,??諕伄惓? if (this->empty()) throw std::exception("Exception stack trace empty!"); --m_top; } bool ExceptionStackTrace::empty() const { return m_top == 0; } int ExceptionStackTrace::top() const { return m_top; } int ExceptionStackTrace::size() const { return m_size; } void ExceptionStackTrace::print_stack_trace() const { // 從后往前,因?yàn)闂5男再|(zhì)后進(jìn)先出 for (int i = m_top - 1; i >= 0; --i) { printf(" At method \"%s\"\n", m_message_trace[i]); } } void ExceptionStackTrace::throw_(SuperException except) { // 拋出一個(gè)異常 printf("Unhandled exception: %s: %s\n", except.exception_name(), except.message()); this->print_stack_trace(); exit(-1); }
二、新異常處理系統(tǒng)中異常的定義
異常包括信息和異常名稱,同時(shí)還需要支持自定義異常。
所以我們可以搞一個(gè)用于新異常系統(tǒng)的父類異常,自定義的異常全部繼承這個(gè)類即可。
父類的名字叫SuperException,下面是類結(jié)構(gòu):
#define M_GetName(data) #data // 宏定義,獲取字符串形式類的名稱 class SuperException { private: const char* m_exception_name = nullptr; // 異常名稱 const char* m_message = nullptr; // 異常消息 public: // 構(gòu)造函數(shù) SuperException(const char* message, const char* exception_name = M_GetName(SuperException)); const char* message() const; // 獲取異常消息 const char* exception_name() const; // 獲取異常名稱 };
然后是實(shí)現(xiàn)部分:
SuperException::SuperException(const char* message, const char* exception_name) { m_message = message; m_exception_name = exception_name; } const char* SuperException::message() const { return m_message; } const char* SuperException::exception_name() const { return m_exception_name; }
具體用法:
int main() { est.push(__FUNCSIG__); int i = 0; scanf_s("%d", &i); if (i == 128) est.throw_(SuperException("這是一個(gè)異常。")); est.pop(); return 0; }
當(dāng)輸入128時(shí):
三、超級(jí)運(yùn)用
#include "ExceptionStackTrace.h" static ExceptionStackTrace est = 128; // 自定義一個(gè)異常 class IndexOutOfBoundsException : public SuperException { public: IndexOutOfBoundsException(const char* message, const char* exception_name = M_GetName(Exception)) : SuperException(message, exception_name) { } }; // 自定義一個(gè)異常 class BadArrayException : public SuperException { public: BadArrayException(const char* message, const char* exception_name = M_GetName(Exception)) : SuperException(message, exception_name) { } }; template<typename T> class Array { private: T* m_array; size_t m_length; private: // 下標(biāo)檢查 void index_check(size_t index) { est.push(__FUNCSIG__); if (index >= m_length) { est.throw_(IndexOutOfBoundsException("Index out of bounds.")); } est.pop(); } public: // 構(gòu)造器 Array(size_t length) { est.push(__FUNCSIG__); m_length = length; try { m_array = new T[length]; } catch (std::bad_alloc) { est.throw_(BadArrayException("Cannot create a Array object because no space.")); } est.pop(); } // 索引訪問(wèn) T& operator[](size_t index) { est.push(__FUNCSIG__); index_check(index); est.pop(); return m_array[index]; } }; int main() { est.push(__FUNCSIG__); Array<void*> a = 128; a[129] = new char[16]; est.pop(); return 0; }
結(jié)果:
為一的遺憾就是沒法加上行號(hào)文件等提示信息,如果能夠?qū)崿F(xiàn)的話,我將會(huì)在下一篇博客中提及。
到此這篇關(guān)于C++實(shí)現(xiàn)并優(yōu)化異常系統(tǒng)的文章就介紹到這了,更多相關(guān)C++異常系統(tǒng)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼
以下是對(duì)歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的介紹,需要的朋友可以過(guò)來(lái)參考下2013-08-08C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)
下面小編就為大家?guī)?lái)一篇C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2016-12-12C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的三子棋項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的三子棋項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-08-08C語(yǔ)言實(shí)現(xiàn)酒店管理系統(tǒng)
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)酒店管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-06-06C++深入詳解單例模式與特殊類設(shè)計(jì)的實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了C++單例模式和特殊類的設(shè)計(jì),單例模式這種類型的設(shè)計(jì)模式屬于創(chuàng)建型模式,它提供了一種創(chuàng)建對(duì)象的最佳方式,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助2022-06-06C語(yǔ)言中返回錯(cuò)誤信息的相關(guān)函數(shù)用法總結(jié)
這篇文章主要介紹了C語(yǔ)言中返回錯(cuò)誤信息的相關(guān)函數(shù)用法總結(jié),包括strerror()函數(shù)和perror()函數(shù)以及ferror()函數(shù)的使用,需要的朋友可以參考下2015-09-09