女人自慰AV免费观看内涵网,日韩国产剧情在线观看网址,神马电影网特片网,最新一级电影欧美,在线观看亚洲欧美日韩,黄色视频在线播放免费观看,ABO涨奶期羡澄,第一导航fulione,美女主播操b

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

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

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

介紹C語言中錯(cuò)誤處理和異常處理的一些常用的方法和策略

Linux愛好者 ? 來源:良許Linux ? 2024-02-28 14:25 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

C語言是一種低級(jí)的、靜態(tài)的、結(jié)構(gòu)化的編程語言,它沒有提供像C++Java等高級(jí)語言中的異常處理機(jī)制,例如try-catch-finally等。

因此,C語言中的錯(cuò)誤處理和異常處理需要采用一些其他的方法和策略,以便在程序運(yùn)行過程中發(fā)現(xiàn)、報(bào)告和處理錯(cuò)誤或異常情況,從而保證程序的正確性和穩(wěn)定性。

本文將介紹C語言中的錯(cuò)誤處理和異常處理的一些常用的方法和策略,以及如何使用setjmp和longjmp這兩個(gè)標(biāo)準(zhǔn)庫函數(shù)來實(shí)現(xiàn)非局部跳轉(zhuǎn),從而在某些情況下模擬異常處理的效果。

錯(cuò)誤處理和異常處理的概念

在討論C語言中的錯(cuò)誤處理和異常處理之前,我們先來區(qū)分一下錯(cuò)誤和異常這兩個(gè)概念。一般來說,錯(cuò)誤是指程序中存在的邏輯或語法上的缺陷,導(dǎo)致程序無法按照預(yù)期的方式運(yùn)行或產(chǎn)生正確的結(jié)果。

例如,數(shù)組越界、空指針解引用、除零操作等都是典型的錯(cuò)誤。錯(cuò)誤通常是可以通過修改代碼來避免或修復(fù)的。

而異常是指程序在運(yùn)行過程中遇到了一些意料之外或無法控制的情況,導(dǎo)致程序無法繼續(xù)正常運(yùn)行或完成預(yù)期的任務(wù)。

例如,文件打開失敗、內(nèi)存分配失敗、信號(hào)中斷等都是典型的異常。異常通常是由于外部環(huán)境或系統(tǒng)資源的變化或限制所引起的,不一定是程序本身的缺陷所導(dǎo)致的。

因此,錯(cuò)誤處理和異常處理有不同的目標(biāo)和方法。錯(cuò)誤處理主要是在編碼階段通過檢查代碼邏輯、使用調(diào)試工具、進(jìn)行單元測試等方式來發(fā)現(xiàn)并消除錯(cuò)誤。

而異常處理主要是在運(yùn)行階段通過檢查函數(shù)返回值、使用信號(hào)處理函數(shù)、設(shè)置錯(cuò)誤處理函數(shù)等方式來捕獲并處理異常。

錯(cuò)誤處理和異常處理的方法和策略

C語言中沒有提供統(tǒng)一的錯(cuò)誤處理和異常處理機(jī)制,但是提供了一些基本的工具和約定,可以根據(jù)不同的情況選擇合適的方法和策略來進(jìn)行錯(cuò)誤處理和異常處理。以下是一些常用的方法和策略:

檢查函數(shù)返回值:這是最常見也最基本的錯(cuò)誤處理和異常處理方法,就是在調(diào)用一個(gè)函數(shù)后,檢查其返回值是否符合預(yù)期或是否表示出錯(cuò)或失敗。如果出錯(cuò)或失敗,則根據(jù)返回值或者全局變量errno(定義在errno.h頭文件中)來判斷出錯(cuò)或失敗的原因,并采取相應(yīng)的措施,例如打印出錯(cuò)信息、釋放資源、返回錯(cuò)誤碼等。例如:

#include 
#include 
#include 

int main() {
    // 打開一個(gè)文件
    FILE *fp = fopen("test.txt", "r");
    // 檢查文件是否打開成功
    if (fp == NULL) {
        // 打印出錯(cuò)信息
        perror("fopen");
        // 返回非零值表示出錯(cuò)
        return 1;
    }
    // 讀取文件內(nèi)容
    char buf[100];
    // 檢查文件是否讀取成功
    if (fgets(buf, 100, fp) == NULL) {
        // 打印出錯(cuò)信息
        perror("fgets");
        // 關(guān)閉文件
        fclose(fp);
        // 返回非零值表示出錯(cuò)
        return 2;
    }
    // 打印文件內(nèi)容
    printf("The content of the file is: %s
", buf);
    // 關(guān)閉文件
    fclose(fp);
    // 返回零值表示成功
    return 0;
}

使用assert宏:這是一種用于調(diào)試階段的錯(cuò)誤處理方法,就是在代碼中插入一些斷言,用于檢查程序的某些假設(shè)或前提是否成立。如果斷言失敗,則表示程序中存在邏輯錯(cuò)誤,程序會(huì)終止并打印出錯(cuò)信息。assert宏定義在assert.h頭文件中,其語法為:

assert(expression);

其中expression是一個(gè)表達(dá)式,如果為真,則繼續(xù)執(zhí)行后面的代碼;如果為假,則終止程序并打印出錯(cuò)信息。例如:

#include 
#include 

int main() {
    // 定義一個(gè)變量
    int x = 10;
    // 斷言x大于0
    assert(x > 0);
    // 打印x的值
    printf("x is %d
", x);
    // 修改x的值
    x = -10;
    // 斷言x大于0
    assert(x > 0);
    // 打印x的值
    printf("x is %d
", x);
    return 0;
}

輸出:

x is 10
Assertion failed: (x > 0), function main, file test.c, line 15.
Abort trap: 6

可以看到,當(dāng)?shù)诙€(gè)斷言失敗時(shí),程序就終止了,并打印了出錯(cuò)信息,包括斷言的表達(dá)式、函數(shù)名、文件名和行號(hào)。這樣可以方便地定位錯(cuò)誤的位置和原因。需要注意的是,assert宏只在調(diào)試階段有效,如果在編譯時(shí)定義了宏NDEBUG,則assert宏會(huì)被忽略,不會(huì)對程序產(chǎn)生任何影響。例如:

gcc -DNDEBUG test.c -o test

這樣編譯后,即使斷言失敗,程序也不會(huì)終止,而是繼續(xù)執(zhí)行后面的代碼。

使用信號(hào)處理函數(shù):這是一種用于處理運(yùn)行時(shí)異常的方法,就是在程序中注冊一些信號(hào)處理函數(shù),用于響應(yīng)系統(tǒng)或用戶發(fā)送的一些信號(hào)。信號(hào)是一種軟件中斷,用于通知進(jìn)程發(fā)生了某些異常或事件。例如,當(dāng)程序試圖訪問非法內(nèi)存地址時(shí),系統(tǒng)會(huì)發(fā)送SIGSEGV信號(hào);當(dāng)用戶按下Ctrl-C鍵時(shí),系統(tǒng)會(huì)發(fā)送SIGINT信號(hào);當(dāng)程序執(zhí)行除零操作時(shí),系統(tǒng)會(huì)發(fā)送SIGFPE信號(hào)等。C語言提供了signal函數(shù)來設(shè)置信號(hào)處理函數(shù),其語法為:

void (*signal(int signum, void (*handler)(int)))(int);

其中signum是要處理的信號(hào)的編號(hào),handler是要設(shè)置的信號(hào)處理函數(shù)的指針。如果handler為SIG_IGN,則表示忽略該信號(hào);如果handler為SIG_DFL,則表示恢復(fù)該信號(hào)的默認(rèn)處理方式。signal函數(shù)返回一個(gè)指針,指向之前設(shè)置的信號(hào)處理函數(shù)。例如:

#include 
#include 

// 定義一個(gè)信號(hào)處理函數(shù)
void handler(int signum) {
    // 打印收到的信號(hào)編號(hào)
    printf("Received signal %d
", signum);
}

int main() {
    // 設(shè)置SIGINT信號(hào)的處理函數(shù)為handler
    signal(SIGINT, handler);
    // 循環(huán)等待用戶輸入
    while (1) {
        char c = getchar();
        // 如果輸入q,則退出循環(huán)
        if (c == 'q') {
            break;
        }
    }
    return 0;
}

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

^CReceived signal 2
^CReceived signal 2
q

可以看到,當(dāng)用戶按下Ctrl-C鍵時(shí),程序不會(huì)終止,而是調(diào)用了自定義的信號(hào)處理函數(shù),并打印了收到的信號(hào)編號(hào)(2表示SIGINT)。當(dāng)用戶輸入q時(shí),程序才退出循環(huán)。

使用setjmp和longjmp函數(shù):這是一種用于實(shí)現(xiàn)非局部跳轉(zhuǎn)的方法,就是在程序中設(shè)置一個(gè)跳轉(zhuǎn)點(diǎn),并在某些情況下跳轉(zhuǎn)到該跳轉(zhuǎn)點(diǎn),從而繞過中間的一些代碼或函數(shù)。這樣可以在某些情況下模擬異常處理的效果,例如在發(fā)生錯(cuò)誤或異常時(shí),直接跳轉(zhuǎn)到錯(cuò)誤處理或資源釋放的代碼,而不需要逐層返回。setjmp和longjmp函數(shù)定義在setjmp.h頭文件中,其語法為:

int setjmp(jmp_buf env);
void longjmp(jmp_buf env, int val);

其中env是一個(gè)用于存儲(chǔ)跳轉(zhuǎn)點(diǎn)信息的數(shù)據(jù)類型,它實(shí)際上是一個(gè)數(shù)組,包含了程序計(jì)數(shù)器、棧指針、寄存器等信息。val是一個(gè)用于傳遞跳轉(zhuǎn)原因的整數(shù)值,它不能為0。setjmp函數(shù)用于設(shè)置跳轉(zhuǎn)點(diǎn),并返回0;longjmp函數(shù)用于跳轉(zhuǎn)到跳轉(zhuǎn)點(diǎn),并使setjmp函數(shù)返回val。例如:

#include 
#include 

// 定義一個(gè)全局的env變量
jmp_buf env;

// 定義一個(gè)可能發(fā)生錯(cuò)誤的函數(shù)
void foo(int x) {
    // 如果x為0,則發(fā)生除零錯(cuò)誤,跳轉(zhuǎn)到env,并傳遞1
    if (x == 0) {
        longjmp(env, 1);
    }
    // 否則,正常執(zhí)行,并打印結(jié)果
    printf("100 / %d = %d
", x, 100 / x);
}

int main() {
    // 設(shè)置跳轉(zhuǎn)點(diǎn),并接收返回值
    int ret = setjmp(env);
    // 如果返回值為0,則表示正常執(zhí)行
    if (ret == 0) {
        // 調(diào)用foo函數(shù),傳入一個(gè)非零值
        foo(10);
        // 調(diào)用foo函數(shù),傳入一個(gè)零值
        foo(0);
    } else {
        // 如果返回值不為0,則表示發(fā)生錯(cuò)誤或異常,根據(jù)返回值打印出錯(cuò)信息
        switch (ret) {
            case 1:
                printf("Error: division by zero
");
                break;
            default:
                printf("Unknown error
");
                break;
        }
    }
    return 0;
}

輸出:

100 / 10 = 10
Error: division by zero

可以看到,當(dāng)調(diào)用foo函數(shù)時(shí),如果傳入的參數(shù)為0,則會(huì)觸發(fā)longjmp函數(shù),從而跳轉(zhuǎn)到setjmp函數(shù)所在的位置,并使setjmp函數(shù)返回1。這樣就可以根據(jù)返回值來判斷發(fā)生了什么錯(cuò)誤或異常,并進(jìn)行相應(yīng)的處理。需要注意的是,使用setjmp和longjmp函數(shù)時(shí)要遵循一些規(guī)則和限制,例如:

不要在setjmp和longjmp之間修改env變量的內(nèi)容。

不要在setjmp和longjmp之間修改任何具有全局或靜態(tài)存儲(chǔ)期的變量。

不要在setjmp和longjmp之間調(diào)用任何可能改變程序狀態(tài)或資源的函數(shù)。

不要在多線程環(huán)境中使用setjmp和longjmp函數(shù)。

總結(jié)

C語言中的錯(cuò)誤處理和異常處理需要采用一些其他的方法和策略,以便在程序運(yùn)行過程中發(fā)現(xiàn)、報(bào)告和處理錯(cuò)誤或異常情況,從而保證程序的正確性和穩(wěn)定性。

本文介紹了C語言中的錯(cuò)誤處理和異常處理的一些常用的方法和策略,以及如何使用setjmp和longjmp函數(shù)來實(shí)現(xiàn)非局部跳轉(zhuǎn),從而在某些情況下模擬異常處理的效果。希望這些內(nèi)容能夠?qū)δ阌兴鶐椭贑語言中更好地進(jìn)行錯(cuò)誤處理和異常處理。




審核編輯:劉清

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

    關(guān)注

    31

    文章

    5434

    瀏覽量

    124533
  • 信號(hào)處理
    +關(guān)注

    關(guān)注

    48

    文章

    1056

    瀏覽量

    104107
  • JAVA
    +關(guān)注

    關(guān)注

    20

    文章

    2989

    瀏覽量

    109790
  • 計(jì)數(shù)器
    +關(guān)注

    關(guān)注

    32

    文章

    2291

    瀏覽量

    96423
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7632

    瀏覽量

    141790

原文標(biāo)題:C語言錯(cuò)誤處理和異常處理方法和策略,如何實(shí)現(xiàn)非局部跳轉(zhuǎn)

文章出處:【微信號(hào):LinuxHub,微信公眾號(hào):Linux愛好者】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

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

掃碼添加小助手

加入工程師交流群

    評(píng)論

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

    嵌入式編程錯(cuò)誤處理機(jī)制設(shè)計(jì)

    本文主要總結(jié)嵌入式系統(tǒng)C語言編程中,主要的錯(cuò)誤處理方式。文中涉及的代碼運(yùn)行環(huán)境如下。
    發(fā)表于 04-28 09:59 ?1033次閱讀
    嵌入式編程<b class='flag-5'>錯(cuò)誤處理</b>機(jī)制設(shè)計(jì)

    嵌入式系統(tǒng)C語言編程中主要的錯(cuò)誤處理方式

    本文主要總結(jié)嵌入式系統(tǒng)C語言編程中,主要的錯(cuò)誤處理方式。
    發(fā)表于 07-24 16:40 ?1274次閱讀
    嵌入式系統(tǒng)<b class='flag-5'>C</b><b class='flag-5'>語言</b>編程中主要的<b class='flag-5'>錯(cuò)誤處理</b>方式

    什么是斷言?C語言中斷言的語法和用法

    在軟件開發(fā)過程中,我們經(jīng)常需要處理各種錯(cuò)誤異常情況。為了提高代碼的健壯性和可靠性,我們需要使用一些工具和技術(shù)來檢測和處理這些問題。本篇博客
    發(fā)表于 08-03 10:34 ?3510次閱讀

    Rust語言中錯(cuò)誤處理的機(jī)制

    在Rust語言中錯(cuò)誤處理項(xiàng)非常重要的任務(wù)。由于Rust語言采用靜態(tài)類型檢查,在編譯時(shí)就能發(fā)現(xiàn)很多潛在的錯(cuò)誤,這使得程序員能夠更加自信和
    的頭像 發(fā)表于 09-19 14:54 ?1899次閱讀

    嵌入式C編程常用異常錯(cuò)誤處理

    嵌入式C編程中,異常錯(cuò)誤處理是確保系統(tǒng)穩(wěn)定性和可靠性的重要部分。以下是一些常見的異常錯(cuò)誤處理
    發(fā)表于 08-06 14:32

    C語言和匯編語言混合編程方法C語言中處理方法

    C語言和匯編語言混合編程方法C語言中處理
    發(fā)表于 01-06 14:36 ?36次下載

    C語言異常處理案例代碼

    相信很多朋友在此之前可能根本沒有使用或者聽說過C語言異常處理,印象中都是C++或者java才有的東西,
    的頭像 發(fā)表于 12-22 08:44 ?4016次閱讀

    嵌入式系統(tǒng)C語言編程中的錯(cuò)誤處理資料總結(jié)

    本文主要總結(jié)嵌入式系統(tǒng)C語言編程中,主要的錯(cuò)誤處理方式。文中涉及的代碼運(yùn)行環(huán)境如下:
    發(fā)表于 11-28 10:39 ?2113次閱讀

    異常處理錯(cuò)誤碼管理

    團(tuán)隊(duì)達(dá)成共識(shí),統(tǒng)規(guī)范就可以。 下面介紹下我使用的處理異常的方式。 自定義異常 創(chuàng)建
    的頭像 發(fā)表于 09-25 14:51 ?1098次閱讀
    <b class='flag-5'>異常</b><b class='flag-5'>處理</b>和<b class='flag-5'>錯(cuò)誤</b>碼管理

    RS232通信時(shí)怎么處理錯(cuò)誤?RS232通信中的錯(cuò)誤處理方法

    錯(cuò)誤是RS232通信中非常重要的個(gè)部分。下面介紹一些處理RS232通信錯(cuò)誤
    的頭像 發(fā)表于 10-17 16:33 ?3890次閱讀

    Service層的異常處理

    般初學(xué)者學(xué)習(xí)編碼和[錯(cuò)誤處理]時(shí),先知道[編程語言]有處理錯(cuò)誤的形式或約定(如Java就拋
    的頭像 發(fā)表于 01-08 11:29 ?833次閱讀

    如何解決C語言中的“訪問權(quán)限沖突”異常C語言引發(fā)異常原因分析

    一些措施來解決和防止其發(fā)生。本文將詳細(xì)介紹C語言中訪問權(quán)限沖突異常的原因以及解決方法
    的頭像 發(fā)表于 01-12 16:03 ?7121次閱讀

    C語言中錯(cuò)誤處理機(jī)制解析

    C 語言不提供對錯(cuò)誤處理的直接支持,但是作為種系統(tǒng)編程語言,它以返回值的形式允許您訪問底層數(shù)據(jù)。
    的頭像 發(fā)表于 02-26 11:19 ?783次閱讀

    socket編程中的錯(cuò)誤處理技巧

    錯(cuò)誤處理能夠確保程序在遇到異常情況時(shí)不會(huì)崩潰,而是能夠優(yōu)雅地處理問題。 提升用戶體驗(yàn) :通過適當(dāng)?shù)?b class='flag-5'>錯(cuò)誤處理,可以給用戶提供清晰的錯(cuò)誤信息
    的頭像 發(fā)表于 11-01 17:47 ?1546次閱讀

    串口通訊異常處理方法 串口設(shè)備連接方式

    等。 軟件檢測 :在軟件層面,可以通過檢查接收到的數(shù)據(jù)幀是否符合預(yù)期的格式和協(xié)議來檢測異常。 2. 錯(cuò)誤處理策略 旦檢測到異常,可以采取以
    的頭像 發(fā)表于 12-27 09:53 ?4085次閱讀