欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++實(shí)現(xiàn)并優(yōu)化異常系統(tǒng)

 更新時(shí)間:2022年08月01日 09:26:20   作者:阿莫·西林  
異常處理是C++的一項(xiàng)語(yǔ)言機(jī)制,用于在程序中處理異常事件,下面這篇文章主要給大家介紹了關(guān)于C++中異常的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下

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)代碼

    歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼

    以下是對(duì)歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的介紹,需要的朋友可以過(guò)來(lái)參考下
    2013-08-08
  • 淺談C++11的std::mem_fn源碼解析

    淺談C++11的std::mem_fn源碼解析

    本文是基于gcc-4.9.0的源代碼進(jìn)行分析,std::mem_fn是C++11才加入標(biāo)準(zhǔn)的,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-06-06
  • C語(yǔ)言分裝實(shí)現(xiàn)三子棋游戲詳解

    C語(yǔ)言分裝實(shí)現(xiàn)三子棋游戲詳解

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言分裝實(shí)現(xiàn)三子棋游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-05-05
  • C++對(duì)象內(nèi)存分布詳解(包括字節(jié)對(duì)齊和虛函數(shù)表)

    C++對(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-12
  • C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的三子棋項(xiàng)目

    C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的三子棋項(xiàng)目

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)簡(jiǎn)單的三子棋項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • C++ continue和break語(yǔ)句

    C++ continue和break語(yǔ)句

    這篇文章主要介紹了C++ continue和break語(yǔ)句,文章圍繞continue和break語(yǔ)句的相關(guān)資料展開詳細(xì)內(nèi)容,需要的朋友可以參考一下,希望對(duì)大家有所幫助
    2021-11-11
  • C++中set的用法學(xué)習(xí)

    C++中set的用法學(xué)習(xí)

    Set是C++?STL(標(biāo)準(zhǔn)模板庫(kù))的一個(gè)容器類,它用于存儲(chǔ)不同的值,并且可以按照特定順序進(jìn)行訪問(wèn)和操作。本文就來(lái)通過(guò)一些示例和大家簡(jiǎn)單講講set的用法吧
    2023-05-05
  • C語(yǔ)言實(shí)現(xiàn)酒店管理系統(tǒng)

    C語(yǔ)言實(shí)現(xiàn)酒店管理系統(tǒng)

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)酒店管理系統(tǒng),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-06-06
  • C++深入詳解單例模式與特殊類設(shè)計(jì)的實(shí)現(xiàn)

    C++深入詳解單例模式與特殊類設(shè)計(jì)的實(shí)現(xiàn)

    這篇文章主要為大家詳細(xì)介紹了C++單例模式和特殊類的設(shè)計(jì),單例模式這種類型的設(shè)計(jì)模式屬于創(chuàng)建型模式,它提供了一種創(chuàng)建對(duì)象的最佳方式,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-06-06
  • C語(yǔ)言中返回錯(cuò)誤信息的相關(guān)函數(shù)用法總結(jié)

    C語(yǔ)言中返回錯(cuò)誤信息的相關(guān)函數(shù)用法總結(jié)

    這篇文章主要介紹了C語(yǔ)言中返回錯(cuò)誤信息的相關(guān)函數(shù)用法總結(jié),包括strerror()函數(shù)和perror()函數(shù)以及ferror()函數(shù)的使用,需要的朋友可以參考下
    2015-09-09

最新評(píng)論