C語言異常處理機(jī)制實(shí)例代碼分析

這篇文章主要介紹了C語言異常處理機(jī)制實(shí)例代碼分析的相關(guān)知識,內(nèi)容詳細(xì)易懂,操作簡單快捷,具有一定借鑒價值,相信大家閱讀完這篇C語言異常處理機(jī)制實(shí)例代碼分析文章都會有所收獲,下面我們一起來看看吧。

創(chuàng)新互聯(lián)建站是一家集網(wǎng)站建設(shè),北碚企業(yè)網(wǎng)站建設(shè),北碚品牌網(wǎng)站建設(shè),網(wǎng)站定制,北碚網(wǎng)站建設(shè)報價,網(wǎng)絡(luò)營銷,網(wǎng)絡(luò)優(yōu)化,北碚網(wǎng)站推廣為一體的創(chuàng)新建站企業(yè),幫助傳統(tǒng)企業(yè)提升企業(yè)形象加強(qiáng)企業(yè)競爭力??沙浞譂M足這一群體相比中小企業(yè)更為豐富、高端、多元的互聯(lián)網(wǎng)需求。同時我們時刻保持專業(yè)、時尚、前沿,時刻以成就客戶成長自我,堅(jiān)持不斷學(xué)習(xí)、思考、沉淀、凈化自己,讓我們?yōu)楦嗟钠髽I(yè)打造出實(shí)用型網(wǎng)站。

異常處理機(jī)制:setjmp()函數(shù)與longjmp()函數(shù)

C標(biāo)準(zhǔn)庫提供兩個特殊的函數(shù):setjmp() 及 longjmp(),這兩個函數(shù)是結(jié)構(gòu)化異常的基礎(chǔ),正是利用這兩個函數(shù)的特性來實(shí)現(xiàn)異常。
所以,異常的處理過程可以描述為這樣:
首先設(shè)置一個跳轉(zhuǎn)點(diǎn)(setjmp() 函數(shù)可以實(shí)現(xiàn)這一功能),然后在其后的代碼中任意地方調(diào)用 longjmp() 跳轉(zhuǎn)回這個跳轉(zhuǎn)點(diǎn)上,以此來實(shí)現(xiàn)當(dāng)發(fā)生異常時,轉(zhuǎn)到處理異常的程序上,在其后的介紹中將介紹如何實(shí)現(xiàn)。
setjmp() 為跳轉(zhuǎn)返回保存現(xiàn)場并為異常提供處理程序,longjmp() 則進(jìn)行跳轉(zhuǎn)(拋出異常),setjmp() 與 longjmp() 可以在函數(shù)間進(jìn)行跳轉(zhuǎn),這就像一個全局的 goto 語句,可以跨函數(shù)跳轉(zhuǎn)。
舉個例子,程序在 main() 函數(shù)內(nèi)使用 setjmp() 設(shè)置跳轉(zhuǎn),并調(diào)用另一函數(shù)A,函數(shù)A內(nèi)調(diào)用B,B拋出異常(調(diào)用longjmp() 函數(shù)),則程序直接跳回到 main() 函數(shù)內(nèi)使用 setjmp() 的地方返回,并且返回一個值。

-------------------------------------------------------------------------------------------------------------------------

jmp_buf 異常結(jié)構(gòu)

使用 setjmp() 及 longjmp() 函數(shù)前,需要先認(rèn)識一下 jmp_buf 異常結(jié)構(gòu)。jmp_buf 將使用在 setjmp() 函數(shù)中,用于保存當(dāng)前程序現(xiàn)場(保存當(dāng)前需要用到的寄存器的值),jmp_buf 結(jié)構(gòu)在 setjmp.h 文件內(nèi)聲明:

typedef struct
{
unsigned j_sp; // 堆棧指針寄存器
unsigned j_ss; // 堆棧段
unsigned j_flag; // 標(biāo)志寄存器
unsigned j_cs; // 代碼段
unsigned j_ip; // 指令指針寄存器
unsigned j_bp; // 基址指針
unsigned j_di; // 目的指針
unsigned j_es; // 附加段
unsigned j_si; // 源變址
unsigned j_ds; // 數(shù)據(jù)段
} jmp_buf;

jmp_buf 結(jié)構(gòu)存放了程序當(dāng)前寄存器的值,以確保使用 longjmp() 后可以跳回到該執(zhí)行點(diǎn)上繼續(xù)執(zhí)行。

-------------------------------------------------------------------------------------------------------------------------

setjmp() 與 longjmp() 函數(shù)詳細(xì)說明

setjmp() 與 longjmp() 函數(shù)原型如下:

void _Cdecl longjmp(jmp_buf jmpb, int retval);

int _Cdecl setjmp(jmp_buf jmpb);

_Cdecl 聲明函數(shù)的參數(shù)使用標(biāo)準(zhǔn)C的進(jìn)棧方式(由右向左)壓棧,_Cdecl 是C語言的一種調(diào)用約定,除此以外,PASCAL 也是調(diào)用約定之一。C標(biāo)準(zhǔn)調(diào)用約定(_Cdecl)所聲明的函數(shù)不自動清除堆棧,這一事務(wù)由調(diào)用者自行負(fù)責(zé)――這也是C可以支持不固定個數(shù)的參數(shù)的原因。此外,這一調(diào)用約定將在函數(shù)名前添加一個下劃線字符,如某一函數(shù)聲明為:

int cdecl DoSomething(void);

編譯時將自動為 DoSomething 加上下劃線前綴,即函數(shù)名變?yōu)椋? _DoSomething。

setjmp() 與 longjmp() 函數(shù)都使用了 jmp_buf 結(jié)構(gòu)作為形參,它們的調(diào)用關(guān)系是這樣的:

首先調(diào)用 setjmp() 函數(shù)來初始化 jmp_buf 結(jié)構(gòu)變量 jmpb,將當(dāng)前CPU中的大部分影響到程序執(zhí)行的寄存器的值存入 jmpb,為 longjmp() 函數(shù)提供跳轉(zhuǎn),setjmp() 函數(shù)是一個有趣的函數(shù),它能返回兩次,它應(yīng)該是所有庫函數(shù)中唯一一個能返回兩次的函數(shù),第一次是初始化時,返回零,第二次遇到 longjmp() 函數(shù)調(diào)用后,longjmp() 函數(shù)使 setjmp() 函數(shù)發(fā)生第二次返回,返回值由 longjmp() 的第二個參數(shù)給出(整型,這時不應(yīng)該再返回零)。

在使用 setjmp() 初始化 jmpb 后,可以其后的程序中任意地方使用 longjmp() 函數(shù)跳轉(zhuǎn)會 setjmp() 函數(shù)的位置,longjmp() 的第一個參數(shù)便是 setjmp() 初始化的 jmpb,若想跳轉(zhuǎn)回剛才設(shè)置的 setjmp() 處,則 longjmp() 函數(shù)的第一個參數(shù)是 setjmp() 所初始化的 jmpb 這個異常,這也說明一件事,即 jmpb 這個異常,一般需要定義為全局變量,否則,若是局部變量,當(dāng)跨函數(shù)調(diào)用時就幾乎無法使用(除非每次遇到函數(shù)調(diào)用都將 jmpb 以參數(shù)傳遞,然而明顯地,是不值得這樣做的);longjmp() 函數(shù)的第二個參數(shù)是傳給 setjmp() 的第二次返回值,這在介紹 setjmp() 函數(shù)時已經(jīng)介紹過。

異常處理過程

先來對比(參考)一下 C++ 的異常處理,C++ 在語言層上便添加了異常處理機(jī)制,使用 try 塊來包含那些可能出現(xiàn)錯誤的代碼,你可以在 try 塊代碼中拋出異常,C++ 使用 throw 來拋出異常。拋出異常后,將轉(zhuǎn)到異常處理程序中執(zhí)行,C++ 使用 catch 塊來包含那些處理異常的代碼,catch 塊可以接收不同類型的異常。需要說明的是,throw 一般不在 try 塊內(nèi)的代碼中拋出異常,try 塊內(nèi)的代碼調(diào)用了別的函數(shù),如函數(shù)A,函數(shù)A 又調(diào)用了函數(shù) B,throw 可以在函數(shù)B中拋出異常,或者更深的函數(shù)調(diào)用層,無論如何,只要有異常拋出,程序?qū)⑥D(zhuǎn)到 catch 處執(zhí)行。

C中如何實(shí)現(xiàn),或者明確地說是模擬這一功能?

下面介紹的是一些簡單的方法。

現(xiàn)在假設(shè) longjmp() 第二個值為1,即 setjmp() 第二次將返回1。我們使用一組簡單的宏來替代 setjmp() 和 longjmp() 以便使用:

首先定義一個全局的異常:

jmp_buf Jump_Buffer;

因?yàn)?setjmp() 第一次調(diào)用初始化后返回0,第二次返回非0,可以這樣定義一個宏使得它功能接近于 C++ 的 try。

#define try if(!setjmp(Jump_Buffer))

當(dāng) setjmp() 函數(shù)第一次0 時,取非為真,則執(zhí)行 try 塊內(nèi)的代碼,如:

try{

Test();

}

當(dāng)因?yàn)檎{(diào)用 longjmp() 拋出異常而導(dǎo)致 setjmp() 第二次返回時(程序?qū)D(zhuǎn)到 setjmp() 函數(shù)處返回,這時,這時應(yīng)該執(zhí)行的是異常處理代碼。longjmp() 使 setjmp() 函數(shù)返回非0,if(!setjmp(JumpBuffer)) 中將值取非則為假,是以,異常處理放在其后應(yīng)該使用一個 else:

#define catch else

如此看起來便跟 C++ 相似了,setjmp() 函數(shù)的第二次返回導(dǎo)致 if() 中表達(dá)式值為假,剛好使 catch 塊得以執(zhí)行,如:

try{

Test();

}catch{

puts("Error");

}

實(shí)現(xiàn)如 C++ 的 throw 語句,事實(shí)上以宏替換 longjmp(jmp_buf, int) 的調(diào)用:

#define throw longjmp(Jump_Buffer, 1)

下面的例程解釋如何使用這些宏:
-------------------------------------------------------------------------------------------------------------------------

#include"stdio.h"
#include"conio.h"
#include"setjmp.h"
jmp_buf Jump_Buffer;
#define try if(!setjmp(Jump_Buffer))
#define catch else
#define throw longjmp(Jump_Buffer,1)
int Test(int T)
{
    if(T>100)
        throw;
    else
          puts("OK.");
    return 0;
}
int Test_T(int T)
{
    Test(T);
    return 0;
}
int main()
{
    int T;
    try{
          puts("Input a value:");
          scanf("%d",&T);
          T++;
          Test_T(T);
      } catch{
          puts("Input Error!");
      }
    getch();
    return 0;
}

關(guān)于“C語言異常處理機(jī)制實(shí)例代碼分析”這篇文章的內(nèi)容就介紹到這里,感謝各位的閱讀!相信大家對“C語言異常處理機(jī)制實(shí)例代碼分析”知識都有一定的了解,大家如果還想學(xué)習(xí)更多知識,歡迎關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

新聞名稱:C語言異常處理機(jī)制實(shí)例代碼分析
網(wǎng)頁路徑:http://muchs.cn/article48/ghspep.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供外貿(mào)網(wǎng)站建設(shè)、用戶體驗(yàn)響應(yīng)式網(wǎng)站、微信小程序、網(wǎng)站建設(shè)、企業(yè)網(wǎng)站制作

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

成都網(wǎng)頁設(shè)計公司