chinese直男口爆体育生外卖, 99久久er热在这里只有精品99, 又色又爽又黄18禁美女裸身无遮挡, gogogo高清免费观看日本电视,私密按摩师高清版在线,人妻视频毛茸茸,91论坛 兴趣闲谈,欧美 亚洲 精品 8区,国产精品久久久久精品免费

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

C語(yǔ)言的異常處理案例代碼

m3eY_edn_china ? 來(lái)源:未知 ? 作者:易水寒 ? 2017-12-22 08:44 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

相信很多朋友在此之前可能根本沒有使用或者聽說(shuō)過(guò)C語(yǔ)言的異常處理,印象中都是C++或者java才有的東西,C語(yǔ)言怎么會(huì)有異常處理呢?

當(dāng)然估計(jì)在大學(xué)出于一般的性的學(xué)習(xí)考試之類的話老師幾乎是不會(huì)提及C語(yǔ)言的異常處理的,那么到底什么是異常處理?C語(yǔ)言中又該如何來(lái)實(shí)現(xiàn)異常處理呢?

那么我們今天就說(shuō)說(shuō)一種典型的實(shí)現(xiàn)C語(yǔ)言異常處理的方法,以setjmp()函數(shù)和longjmp()函數(shù)實(shí)現(xiàn)的異常處理,我盡可能的把它們是怎樣實(shí)現(xiàn)異常處理方法講解清楚,希望接下來(lái)的內(nèi)容對(duì)你有所幫助,讓你學(xué)到一些新的東西。

首先我們來(lái)了解下異常處理,異常是一個(gè)在程序執(zhí)行期間發(fā)生的事件,它中斷正在執(zhí)行的程序的正常的指令流,而我們的異常處理功能提供了處理程序運(yùn)行時(shí)出現(xiàn)的任何意外或異常情況的方法。

接下來(lái)我們先看看setjmp()函數(shù)和longjmp()函數(shù)實(shí)現(xiàn)C語(yǔ)言異常處理。

setjmp()函數(shù)原型:

int ( jmp_buf env );

如果我們打開源代碼會(huì)發(fā)現(xiàn)在setjmp()函數(shù)中涉及到很多的寄存器的操作,如Ebp、Ebx、Edi、Esi、Esp、 Eip等等,在此就不一一例舉了,我們無(wú)非是想向讀者說(shuō)明一個(gè)問題,那就是在調(diào)用setjmp()函數(shù)的過(guò)程中保存程序的當(dāng)前運(yùn)行時(shí)的堆棧環(huán)境,保存這些堆棧環(huán)境有什么用呢?接下來(lái)我們看看longjmp()函數(shù)。

longjmp()函數(shù)原型:

void longjmp( jmp_buf env, int value );

剛剛上面的函數(shù)功能是保存程序執(zhí)行時(shí)候的堆棧環(huán)境,我們發(fā)現(xiàn)在longjmp()函數(shù)里也有一個(gè)jmp_buf類型的env變量,這其實(shí)是為了保證接下來(lái)調(diào)用longjmp時(shí),會(huì)根據(jù)這個(gè)曾經(jīng)保存的變量來(lái)恢復(fù)先前的環(huán)境,并且當(dāng)前的程序控制流,會(huì)因此而返回到最初調(diào)用setjmp()函數(shù)時(shí)的程序執(zhí)行點(diǎn)。此時(shí),在接下來(lái)的控制流的例程中,所能訪問的所有的變量,包含了longjmp函數(shù)調(diào)用時(shí)所擁有的變量。我們就這樣說(shuō)讀者可能就得有點(diǎn)抽象了,那我們還是來(lái)看看一段代碼后再來(lái)分析吧,在此特地給出了一個(gè)簡(jiǎn)單的代碼,由易到難的來(lái)分析。

[cpp] view plaincopy

#include

#include

jmp_buf buf;

void error_code(void)

{

longjmp(buf,1);

}

int main()

{

double a,b;

printf("請(qǐng)輸入被除數(shù):");

scanf("%lf",&a);

printf("請(qǐng)輸入除數(shù):");

if(setjmp(buf)==0)

{

scanf("%lf",&b);

if(0==b)

error_code();

printf("相除的結(jié)果為:%f\n",a/b);

}

else

printf("出現(xiàn)錯(cuò)誤除數(shù)為0\n");

return 0;

}

運(yùn)行結(jié)果為:

[cpp] view plaincopy

請(qǐng)輸入被除數(shù):12

請(qǐng)輸入除數(shù):0

出現(xiàn)錯(cuò)誤除數(shù)為0

Press any key to continue

看了上面的運(yùn)行結(jié)果,現(xiàn)在我們接著上面的講,在一開始的部分我們并沒有具體的交代setjmp()函數(shù)和longjmp()函數(shù)的返回值和參數(shù)的具體含義。兩個(gè)函數(shù)中的env變量保存的是調(diào)用setjmp()函數(shù)的時(shí)候當(dāng)前運(yùn)行程序的堆棧信息,而longjmp()函數(shù)的調(diào)用就是根據(jù)在調(diào)用setjmp()函數(shù)的時(shí)候的堆棧信息返回到最初調(diào)用setjmp()函數(shù)的地方,而其中的第二個(gè)參數(shù)就是此刻setjmp()函數(shù)的返回值,但是值得注意的就是調(diào)用longjmp()函數(shù)之后setjmp函數(shù)返回的值必須是非零值,如果longjmp傳送的value參數(shù)值為0,那么實(shí)際上setjmp返回的值是1。一開始我們調(diào)用setjmp()函數(shù)的時(shí)候,它的返回值為0,之后再調(diào)用longjmp()函數(shù)的時(shí)候,通過(guò)設(shè)定longjmp()函數(shù)的第二個(gè)參數(shù)來(lái)設(shè)定它的返回值。

現(xiàn)在我們來(lái)分析上邊的代碼,在main()函數(shù)中,我們最初調(diào)用setjmp()函數(shù)的時(shí)候,把當(dāng)前的環(huán)境信息保存在了buf中,函數(shù)返回0,然后往下運(yùn)行,我們輸入0。通過(guò)if語(yǔ)句發(fā)現(xiàn)b的值為0那么就調(diào)用error_code()函數(shù)來(lái)進(jìn)行處理,在該函數(shù)中我們使用了longjmp()函數(shù),其使用方式為longjmp(buf,1);,通過(guò)上面的講解,我們知道第一個(gè)參數(shù)的作用是用來(lái)得到最初調(diào)用setjmp()函數(shù)是的環(huán)境信息,以便在使用longjmp()函數(shù)的時(shí)候能夠正確的返回到setjmp()函數(shù)最初的調(diào)用處,而后面的參數(shù)表示的返回到setjmp()函數(shù)的時(shí)候的返回值。我們?cè)诖朔祷?,所以執(zhí)行else部分的語(yǔ)句。

分析完了上面的代碼,讀者應(yīng)該都知道了兩個(gè)函數(shù)的使用方法,值得注意的地方就是我們?cè)趕etjmp與longjmp結(jié)合使用時(shí),它們必須有嚴(yán)格的先后執(zhí)行順序,先調(diào)用setjmp函數(shù),之后再調(diào)用longjmp函數(shù),以恢復(fù)到先前被保存的“程序執(zhí)行點(diǎn)”。否則,假如在setjmp調(diào)用之前,執(zhí)行l(wèi)ongjmp函數(shù),將導(dǎo)致程序的執(zhí)行流變的不可猜測(cè),很輕易導(dǎo)致程序崩潰而退出。為了加深讀者的對(duì)于兩個(gè)函數(shù)參數(shù)的使用,我們看看下面的代碼:

[cpp] view plaincopy

#include

#include

#include

#include

jmp_buf buf;

void func1()

{

longjmp(buf,1);

}

void func2()

{

longjmp(buf,2);

}

void func3()

{

longjmp(buf,3);

}

int main( void )

{

int value;

char str[50];

value = setjmp( buf );

if( value == 0 )

{

func1();

}

switch( value )

{

case 1:

strcpy( str, "func1 return value" );

break;

case 2:

strcpy( str, "func2 return value" );

break;

case 3:

strcpy( str, "func3 return value" );

break;

default:

strcpy( str, "Other error value" );

break;

}

printf("%s:%d\n",str,value);

if(1==value)

{

func2();

}

if(2==value)

{

func3();

}

return 0;

}

運(yùn)行結(jié)果為:

[cpp] view plaincopy

func1 return value:1

func2 return value:2

func3 return value:3

Press any key to continue

看看運(yùn)行結(jié)果,我們分析下代碼,在每個(gè)函數(shù)中我們調(diào)用longjmp()函數(shù),通過(guò)設(shè)置第二個(gè)參數(shù)為不同的值來(lái)改變setjmp()函數(shù)的返回值,然后我們通過(guò)判斷value值來(lái)打印出是那個(gè)函數(shù)的返回值,我們?cè)诖死e這個(gè)簡(jiǎn)單的代碼是要大家加深對(duì)于這兩個(gè)函數(shù)的參數(shù)的使用情況。如果我們?cè)谏厦娴拇a中稍作修改,在setjmp()函數(shù)的調(diào)用之前調(diào)用longjmp()函數(shù),我們發(fā)現(xiàn)此時(shí)沒有任何的輸出,程序直接崩潰掉退出了。

接下來(lái)我們來(lái)看看一個(gè)函數(shù)的使用,如果對(duì)于這個(gè)函數(shù)不理解的讀者,可以多看幾次我給出的模擬該函數(shù)的實(shí)現(xiàn)代碼。

頭文件: #include

功能:設(shè)置某一信號(hào)的對(duì)應(yīng)動(dòng)作

函數(shù)原型:void (*signal(int signum,void(* handler)(int)))(int);

注意:第一個(gè)參數(shù)signum指明了所要處理的信號(hào)類型,它可以取除了SIGKILL和SIGSTOP外的任何一種信號(hào)。

如果讀者是第一場(chǎng)接觸上面的函數(shù)的話可能有些不知道該如何著手,一時(shí)間有些難以理解,不知道到底是什么意思。別急,我們現(xiàn)在來(lái)逐一分析它到底是什么意思,我們?cè)谥v解之前再來(lái)看看它的另外一種表示方法。

typedef void(*sig_t) ( int );

sig_t signal(int signum,sig_t handler);

把上面的函數(shù)原型拆分為了如上兩行代碼,現(xiàn)在我們分析下上面的兩行代碼。

第一行代碼定義了一個(gè)函數(shù)指針(注:如果有對(duì)函數(shù)指針知識(shí)點(diǎn)不熟悉的讀者可以去閱讀我之前寫的那篇文章《C語(yǔ)言的那些小秘密之函數(shù)指針》),其類型為含有一個(gè)int型參數(shù),無(wú)返回值;

第二行代碼中,signal函數(shù)的返回值是一個(gè)函數(shù)指針,與第一行我們定義的類型相同,第二個(gè)參數(shù)也為一個(gè)函數(shù)指針,其實(shí)signal的返回值就是第二個(gè)函數(shù)指針指向的函數(shù)地址。這樣說(shuō)可能有不少讀者都有些懵的感覺,還是老方法,代碼最有說(shuō)服力,我們還是為讀者模擬下signal的實(shí)現(xiàn)方式,呈現(xiàn)出一段代碼來(lái)分析下。

[cpp] view plaincopy

#include

#include

typedef void (*pfun) ();

pfun signal_call(int a,pfun fdsa);

pfun signal_call(int a,pfun fdsa)

{

return fdsa;

}

void func()

{

printf("hello world!!!\n");

}

int main()

{

pfun p = func;

signal_call(1,p)();

return 0;

}

運(yùn)行結(jié)果為:

[cpp] view plaincopy

hello world!!!

Press any key to continue

現(xiàn)在我們來(lái)分析下上面的代碼,我們采用上面的定義形式實(shí)現(xiàn)了如下兩行代碼:

typedef void (*pfun) ();

pfun signal_call(int a,pfun fdsa);

在接下來(lái)的main()函數(shù)中我們定義了一個(gè)函數(shù)指針p,使其指向了 func()函數(shù),接下來(lái)我們使用了一句 signal_call(1,p)();代碼,實(shí)現(xiàn)了func函數(shù)調(diào)用,那么這到底是怎么實(shí)現(xiàn)的呢?那么我們來(lái)分析下,前面的signal_call(1,p)返回的是一個(gè)函數(shù)指針,在代碼中我們發(fā)現(xiàn)其實(shí)返回的就是p,所以signal_call(1,p)();就可以變形為p(),看到這種形式我們這就可以很清楚的看出,它調(diào)用的就是我們代碼中的func()函數(shù)了?,F(xiàn)在讀者明白了signal()函數(shù)的實(shí)現(xiàn)方法,接下來(lái)我們來(lái)看看一段使用signal捕捉除數(shù)為0時(shí)候的異常代碼。

cpp] view plaincopy

#include

#include

#include

#include

#include

#include

jmp_buf buf;

int err;

void handler( int num )

{

err = num;

printf( "發(fā)生浮點(diǎn)計(jì)算異常\n");

longjmp( buf, 1);

}

int main( void )

{

double a, b;

char str[20];

int ret;

_control87( 0, _MCW_EM );

if( signal( SIGFPE, handler ) == SIG_ERR )

{

printf("綁定失敗\n" );

abort();

}

ret = setjmp( buf );

if(0 == ret )

{

printf("請(qǐng)輸入被除數(shù):");

scanf("%lf",&a);

printf("請(qǐng)輸入除數(shù):");

scanf("%lf",&b);

printf( "a / b = %4.3g\n", a/b);

printf("發(fā)生異常時(shí)候不會(huì)被執(zhí)行的語(yǔ)句\n");

}

return 0;

}

沒有發(fā)生異常時(shí)候的運(yùn)行結(jié)果:

[cpp] view plaincopy

請(qǐng)輸入被除數(shù):123

請(qǐng)輸入除數(shù):3

a / b = 41

發(fā)生異常時(shí)候不會(huì)被執(zhí)行的語(yǔ)句

Press any key to continue

發(fā)生異常時(shí)候的運(yùn)行結(jié)果:

[cpp] view plaincopy

請(qǐng)輸入被除數(shù):12

請(qǐng)輸入除數(shù):0

發(fā)生浮點(diǎn)計(jì)算異常

Press any key to continue

現(xiàn)在來(lái)分析下上面的運(yùn)行結(jié)果,先看看_control87( 0, _MCW_EM );這句,可能很多讀者對(duì)于這代碼比較陌生,它的功能是開啟所有的浮點(diǎn)計(jì)算異常,通常情況下浮點(diǎn)計(jì)算異常是被屏蔽掉的,我們?yōu)榱四軌蚴沟媒酉聛?lái)的signal能夠捕捉到浮點(diǎn)計(jì)算異常,所以要將其開啟。在往下看我們通過(guò)signal( SIGFPE, handler )來(lái)綁定了一個(gè)浮點(diǎn)計(jì)算異常處理函數(shù),如果發(fā)生異常時(shí),那么就調(diào)用handler()函數(shù)來(lái)處理。接下來(lái)通過(guò)ret = setjmp( buf );保存程序運(yùn)行的環(huán)境信息,以便接下來(lái)的調(diào)用longjmp()函數(shù)能夠根據(jù)這個(gè)保存的信息返回該程序先前setjmp()函數(shù)的執(zhí)行點(diǎn)。同時(shí)我們對(duì)比兩次運(yùn)行的結(jié)果發(fā)現(xiàn)如果發(fā)現(xiàn)異常的時(shí)候接下來(lái)的打印語(yǔ)句“printf("發(fā)生異常時(shí)候不會(huì)被執(zhí)行的語(yǔ)句\n");”是不會(huì)被執(zhí)行的,直接跳轉(zhuǎn)到我們綁定的handler()函數(shù)執(zhí)行了,當(dāng)然我們?cè)诖藘H僅是例舉一些簡(jiǎn)單的代碼教會(huì)讀者學(xué)會(huì)使用setjmp()函數(shù)和longjmp()函數(shù)來(lái)實(shí)現(xiàn)異常處理,讀者完全可以在此基礎(chǔ)上編寫出復(fù)雜的異常處理。


聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • C語(yǔ)言
    +關(guān)注

    關(guān)注

    183

    文章

    7642

    瀏覽量

    145116
  • 異常
    +關(guān)注

    關(guān)注

    0

    文章

    23

    瀏覽量

    9487

原文標(biāo)題:嵌入式C小秘密之你不知道的異常處理

文章出處:【微信號(hào):edn-china,微信公眾號(hào):EDN電子技術(shù)設(shè)計(jì)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    c語(yǔ)言中的代碼優(yōu)化

    。不少高級(jí)語(yǔ)言,包括C++,并不對(duì)產(chǎn)生的浮點(diǎn)表達(dá)式從新排序,由于那是一個(gè)至關(guān)復(fù)雜的過(guò)程。須要注意的是,重排序的代碼和原來(lái)的代碼代碼上一致并
    發(fā)表于 01-12 09:45

    C語(yǔ)言主要特點(diǎn)

    ,源程序短,因此輸入程序時(shí)工作量少。 2.運(yùn)算符豐富,C語(yǔ)言的運(yùn)算符包含的范圍很廣泛,共有34種運(yùn)算符。C語(yǔ)言把括號(hào)、賦值和強(qiáng)制類型轉(zhuǎn)換等都作為運(yùn)算符
    發(fā)表于 01-05 07:41

    請(qǐng)問C語(yǔ)言中整形溢出會(huì)產(chǎn)生哪些異常行為?

    C語(yǔ)言中整形溢出會(huì)產(chǎn)生哪些異常行為?
    發(fā)表于 12-26 07:05

    使用setjmp及l(fā)ongjmp函數(shù)處理異常

    使用setjmp和longjmp函數(shù):這是一種用于實(shí)現(xiàn)非局部跳轉(zhuǎn)的方法,就是在程序中設(shè)置一個(gè)跳轉(zhuǎn)點(diǎn),并在某些情況下跳轉(zhuǎn)到該跳轉(zhuǎn)點(diǎn),從而繞過(guò)中間的一些代碼或函數(shù)。這樣可以在某些情況下模擬異常處理的效果
    發(fā)表于 12-11 08:00

    C語(yǔ)言C++之間的區(qū)別是什么

    處理的問題時(shí)拋出異常,并在其他地方捕獲和處理這些異常。 C語(yǔ)言沒有內(nèi)建的
    發(fā)表于 12-11 06:23

    C++程序異常處理機(jī)制

    運(yùn)行代碼進(jìn)行分離,使得程序更加模塊化;另一方面,C++的異常處理可以不需要異常處理
    發(fā)表于 12-02 07:12

    C語(yǔ)言的編程技巧

    一個(gè)成員是一個(gè)未知大小的數(shù)組,適用于動(dòng)態(tài)分配內(nèi)存并關(guān)聯(lián)一個(gè)可變長(zhǎng)度的數(shù)組。? ?3、匿名結(jié)構(gòu)體和聯(lián)合體?:C語(yǔ)言允許在結(jié)構(gòu)體或聯(lián)合體中定義不帶標(biāo)簽的內(nèi)部結(jié)構(gòu)體或聯(lián)合體,簡(jiǎn)化代碼結(jié)構(gòu)。 ?4
    發(fā)表于 11-27 06:46

    C語(yǔ)言的分支結(jié)構(gòu)介紹

    1.簡(jiǎn)單if語(yǔ)句 C語(yǔ)言中的分支結(jié)構(gòu)語(yǔ)句中的if條件語(yǔ)句。 簡(jiǎn)單if語(yǔ)句的基本結(jié)構(gòu)如下: 代碼語(yǔ)言:javascript if(表達(dá)式) { 執(zhí)行
    發(fā)表于 11-25 07:48

    C語(yǔ)言特性

    的執(zhí)行效率和資源利用率有著嚴(yán)苛的要求。C 語(yǔ)言生成的代碼簡(jiǎn)潔緊湊,能夠在有限的硬件條件下快速運(yùn)行,滿足嵌入式系統(tǒng)對(duì)性能的高要求。例如,在智能家居設(shè)備的控制芯片中,C
    發(fā)表于 11-24 07:01

    線路保護(hù)光纖通道異常處理方法

    通道異常的 常見原因、處理步驟及預(yù)防措施 ,幫助運(yùn)維人員快速定位問題,提升故障處理效率。 廣州郵科光纖線路保護(hù)系統(tǒng) 一、光纖通道異常的常見表現(xiàn) 當(dāng)線路保護(hù)光纖通道出現(xiàn)
    的頭像 發(fā)表于 11-17 10:01 ?761次閱讀
    線路保護(hù)光纖通道<b class='flag-5'>異常</b><b class='flag-5'>處理</b>方法

    C語(yǔ)言和單片機(jī)C語(yǔ)言有什么差異

    的目標(biāo)代碼短、運(yùn)行速度高、存儲(chǔ)空間小、符合C語(yǔ)言的ANSI標(biāo)準(zhǔn),生成的代碼遵循Intel目標(biāo)文件格式,而且可與A51匯編語(yǔ)言PL/M51
    發(fā)表于 11-14 07:55

    深入理解C語(yǔ)言C語(yǔ)言循環(huán)控制

    C語(yǔ)言編程中,循環(huán)結(jié)構(gòu)是至關(guān)重要的,它可以讓程序重復(fù)執(zhí)行特定的代碼塊,從而提高編程效率。然而,為了避免程序進(jìn)入無(wú)限循環(huán),C語(yǔ)言提供了多種循
    的頭像 發(fā)表于 04-29 18:49 ?1904次閱讀
    深入理解<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>:<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>循環(huán)控制

    必看!15個(gè)C語(yǔ)言常見陷阱及避坑指南

    ? C語(yǔ)言雖強(qiáng)大,但隱藏的“坑”也不少!稍不留神就會(huì)導(dǎo)致程序崩潰、數(shù)據(jù)異常。本文整理15個(gè)高頻陷阱,助你寫出更穩(wěn)健的代碼! ? 陷阱1:運(yùn)算符優(yōu)先級(jí)混淆? 問題:運(yùn)算符優(yōu)先級(jí)不同可能導(dǎo)
    的頭像 發(fā)表于 03-16 12:10 ?1630次閱讀

    代碼加密、源代碼防泄漏c/c++與git服務(wù)器開發(fā)環(huán)境

    代碼加密對(duì)于很多研發(fā)性單位來(lái)說(shuō)是至關(guān)重要的,當(dāng)然每家企業(yè)的業(yè)務(wù)需求不同所用的開發(fā)環(huán)境及開發(fā)語(yǔ)言也不盡相同,今天主要來(lái)講一下c++及git開發(fā)環(huán)境的源代碼防泄密保護(hù)方案。企業(yè)源
    的頭像 發(fā)表于 02-12 15:26 ?1022次閱讀
    源<b class='flag-5'>代碼</b>加密、源<b class='flag-5'>代碼</b>防泄漏<b class='flag-5'>c</b>/<b class='flag-5'>c</b>++與git服務(wù)器開發(fā)環(huán)境

    分析C語(yǔ)言代碼結(jié)構(gòu)的設(shè)計(jì)問題

    來(lái)分析一個(gè)C語(yǔ)言代碼結(jié)構(gòu)的設(shè)計(jì)問題。 這段代碼,使用了兩次malloc,分別給 p1 和 p2 申請(qǐng)了內(nèi)存。用完后,內(nèi)存釋放,防止內(nèi)存泄漏。 大家覺得,這樣的
    的頭像 發(fā)表于 02-11 09:31 ?762次閱讀