精品国产人成在线_亚洲高清无码在线观看_国产在线视频国产永久2021_国产AV综合第一页一个的一区免费影院黑人_最近中文字幕MV高清在线视频

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

C語言函數宏的三種封裝方式

STM32嵌入式開發 ? 來源:CSDN技術社區 ? 2023-02-13 14:21 ? 次閱讀

1 函數宏介紹

函數宏,即包含多條語句的宏定義,其通常為某一被頻繁調用的功能的語句封裝,且不想通過函數方式封裝來降低額外的彈棧壓棧開銷。

函數宏本質上為宏,可以直接進行定義,例如:


#define INT_SWAP(a,b) 
    int tmp = a;    
    a = b;          
    b = tmp
但上述的宏具有一個明顯的缺點:當遇到if 、while 等語句且不使用花括號僅調用宏時,實際作用范圍在宏的第一個分號后便結束。即a = b 和b = tmp 均不受控制語句所作用。

因此,在工程中,一般使用三種方式來對函數宏進行封裝,分別為{} 、do{...}while(0) 和({}) 。下文將一一對三種方式進行分析,比較各自的優劣點。

2 { } 方式

INT_SWAP 宏使用{} 封裝后形態如下:

#define INT_SWAP(a,b)
{                   
    int tmp = a;    
    a = b;          
    b = tmp;        
}
此時,直接調用與在無花括號的控制語句(如 if、while)中調用均能正常運行,例如:

#define INT_SWAP(a,b) 
{                   
    int tmp = a;    
    a = b;          
    b = tmp;        
}


int main()
{
    int var_a = 1;
    int var_b = 2;


    INT_SWAP(var_a, var_b);
    printf("var_a = %d, var_b = %d
", var_a, var_b);   // var_a = 2, var_b = 1


    if (1)
       INT_SWAP(var_a, var_b);
    printf("var_a = %d, var_b = %d
", var_a, var_b);   // var_a = 1, var_b = 2
}
但當無花括號的 if 語句存在其他分支(else if、else 等)如:

if (1)
   INT_SWAP(var_a, var_b);
else
    printf("hello world!
");
會發現編譯出錯:

...
/mnt/hgfs/share/pr_c/src/main.c: In function ‘main’:
/mnt/hgfs/share/pr_c/src/main.c2: error: ‘else’ without a previous ‘if’
  else
這是因為INT_SWAP(var_a, var_b); 最后的; 已經把if 的作用域終結了,后續的else 當然沒有找到與之匹配的if 了。

因此,解決方法有兩種,分別為不使用;(port.1)或規定必須使用帶花括號的if (port.2),例如:

/* port.1 */
if (1)
   INT_SWAP(var_a, var_b)
else
{
    printf("hello world!
");
}


/* port.2 */
if (1)
{
   INT_SWAP(var_a, var_b);
}
else
{
    printf("hello world!
");
}
可見,不使用 ; 的調用方式無論從程序閱讀還是使用方法方面都是十分別扭的;而規定必須使用帶花括號的 if 的調用方式有違常理的,因為宏函數應該適用于任何語法。

優缺點總結:

優點:簡單粗暴。


缺點:不能在無花括號且有分支的 if 語句中直接調用;能夠不帶 ; 直接調用。

3 do{...}while(0) 方式 INT_SWAP 宏使用 do{...}while(0) 封裝后形態如下:


#define INT_SWAP(a,b)   
do{                     
    int tmp = a;        
    a = b;              
    b = tmp;            
}while(0)
do{...}while(0) 表示只執行一遍 {} 內的語句,表象來說與{} 的功能是一致的。不同的是,do{...}while(0) 可以提前退出函數宏、整合為一條語句與強制調用時必須使用; 。

由于do{...}while(0) 實際為 while 循環,因此可以使用關鍵字break 提前結束循環。利用該特性,可以為函數宏添加參數檢測。例如:

#define INT_SWAP(a,b)  
do{                 
    if (a < 0 || b < 0) 
        break;   
    int tmp = a;     
    a = b;           
    b = tmp;         
}while(0)
由于 do{...}while(0); 實際為一種語法,編譯器會把 do{...}while(0); 認為為一條語句。因此,do{...}while(0) 方式的函數宏可以在無花括號且有分支的if 語句中直接調用。

例如:

#define INT_SWAP(a,b)  
do{                 
    if (a < 0 || b < 0) 
        break;   
    int tmp = a;     
    a = b;           
    b = tmp;         
}while(0)


int main()
{
    int var_a = 1;
    int var_b = 2;


    if (1)
       INT_SWAP(var_a, var_b);
    else
        printf("hello world!
"); 
    printf("var_a = %d, var_b = %d
", var_a, var_b); // var_a = 2, var_b = 1


    return 0;
}
C 語言規定,do{...}while(0) 語法必須使用 ; 作為語句結尾。因此不可能存在以下語句的程序出現:

if (1)
   INT_SWAP(var_a, var_b)
else
{
    printf("hello world!
"); 
}
優缺點總結:

優點:支持在無花括號且有分支的 if 語句中直接調用;支持提前退出函數宏;強制調用時必須使用。 缺點:無返回值,不能作為表達式的右值使用。

4 ({ }) 方式

({}) 為 GNU C 擴展的語法,非 C 語言的原生語法。

INT_SWAP 宏使用({}) 封裝后形態如下:


#define INT_SWAP(a,b)   
({                      
    int tmp = a;        
    a = b;              
    b = tmp;            
})
與do{...}while(0) 相同,({}) 支持在無花括號且有分支的if 語句中直接調用。例如:

#define INT_SWAP(a,b)  
({                 
    int tmp = a;    
    a = b;          
    b = tmp;        
})


int main()
{
    int var_a = 1;
    int var_b = 2;


    if (1)
       INT_SWAP(var_a, var_b);
    else
        printf("hello world!
");
    printf("var_a = %d, var_b = %d
", var_a, var_b); // var_a = 2, var_b = 1


    return 0;
}
與do{...}while(0) 不同的是,({}) 不能提前退出函數宏與支持返回值。({}) 畢竟不是 while 循環,不能直接使用break 退出函數宏是比較容易理解。那支持返回值是什么意思呢?

答案是 C 語言規定({}) 中的最后一條語句的結果為該雙括號體的返回值。

例如:

int main()
{
    int a = ({
        10;
        1000;
    });
    printf("a = %d
", a);    // a = 1000
}
因此,({}) 可以為函數宏提供返回值。 例如:

#define INT_SWAP(a,b)  
({                 
    int ret = 0;  
    if (a < 0 || b < 0) 
    {     
        ret = -1;  
    }     
    else    
    {     
        int tmp = a;    
        a = b;          
        b = tmp;        
    }     
    ret;    
})


int main()
{
    int var_a = 1;
    int var_b = 2;


    if (INT_SWAP(var_a, var_b) != -1)
        printf("swap success !!
");     // swap success !!
    else
        printf("swap fail !!
"); 
    printf("var_a = %d, var_b = %d
", var_a, var_b); // var_a = 2, var_b = 1


    return 0;
}
可見,此時的 INT_SWAP 宏已與函數十分接近。

優缺點總結:

優點:支持在無花括號且有分支的 if 語句中直接調用;有返回值,支持作為表達式的右值。


缺點:不支持提前退出函數宏;非 C 的原生語法,編譯器可能不支持。

5 總結

綜上,在{} 、do{...}while(0) 和 ({}) 這三種函數宏的封裝方式之中,應盡可能不使用{} ,考慮兼容性一般選擇使用do{...}while(0) ,當需要函數宏返回時可以考慮使用({}) 或直接定義函數。

審核編輯:湯梓紅

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • 封裝
    +關注

    關注

    126

    文章

    7783

    瀏覽量

    142723
  • C語言
    +關注

    關注

    180

    文章

    7598

    瀏覽量

    136192
  • 函數
    +關注

    關注

    3

    文章

    4306

    瀏覽量

    62430
  • 編譯
    +關注

    關注

    0

    文章

    654

    瀏覽量

    32806

原文標題:C語言函數宏的三種封裝方式

文章出處:【微信號:c-stm32,微信公眾號:STM32嵌入式開發】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    C語言定義的應用

    C語言中,定義是一預處理指令,用于在代碼中定義和使用常量、函數或代碼片段的替代。
    發表于 08-17 15:33 ?755次閱讀

    C語言中的是什么

    第五章 性能優化5.1 使用定義  在C語言中,是產生內嵌代碼的唯一方法。對于嵌入式系統而言,為了能達到性能要求,是一
    發表于 12-15 08:20

    內聯函數定義的區別介紹

    定義是C語言提供的三種預處理功能的其中一,這三種預處理包括:
    發表于 12-15 15:33 ?2332次閱讀
    內聯<b class='flag-5'>函數</b>和<b class='flag-5'>宏</b>定義的區別介紹

    幾種特殊的函數封裝方式介紹

    : #defineINT_SWAP(a,b) inttmp=a; a=b; b=tmp但上述的具有一個明顯的缺點:當遇到 if、while 等語句且不使用花括號僅調用時,實際作用范圍在的第一個分號后便結束。即 a = b
    的頭像 發表于 11-03 16:03 ?2084次閱讀

    函數三種封裝方式

    范圍在的第一個分號后便結束。即 a = b 和 b = tmp 均不受控制語句所作用。 因此,在工程中,一般使用三種方式
    的頭像 發表于 12-22 15:49 ?3657次閱讀

    如何應用伺服電機的三種控制方式

    一般伺服都有三種控制方式:速度控制方式,轉矩控制方式,位置控制方式。大多數人想知道的就是這三種
    發表于 01-22 06:30 ?7次下載
    如何應用伺服電機的<b class='flag-5'>三種</b>控制<b class='flag-5'>方式</b>

    C語言訪問MCU寄存器的三種方式

    C語言訪問MCU寄存器的三種方式 MCU中的特殊功能寄存器SFR,實際上就是SRAM地址已經確定的SRAM單元,在C
    發表于 10-28 17:21 ?11次下載
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>訪問MCU寄存器的<b class='flag-5'>三種</b><b class='flag-5'>方式</b>

    C語言定義與預處理、函數函數

    目錄前言一、C語言預處理二、定義、函數四、函數庫五、自己制作靜態鏈接庫(ubuntu 環境下
    發表于 12-07 21:06 ?2次下載
    <b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>宏</b>定義與預處理、<b class='flag-5'>函數</b>和<b class='flag-5'>函數</b>庫

    C語言函數調用的三種方式

    C 語言是一門面向過程的編程語言,通過一個又一個函數,把計算、過程控制等邏輯,包裝成一個個獨立的處理單元。
    發表于 10-08 14:18 ?5513次閱讀

    常見函數三種封裝方式

    函數,即包含多條語句的定義,其通常為某一被頻繁調用的功能的語句封裝,且不想通過函數方式
    的頭像 發表于 05-05 15:29 ?493次閱讀

    幾種特殊的函數封裝方式,你會嗎?

    函數,即包含多條語句的定義,其通常為某一被頻繁調用的功能的語句封裝,且不想通過函數方式
    的頭像 發表于 09-13 15:06 ?577次閱讀

    C語言函數封裝技巧分享

    函數,即包含多條語句的定義,其通常為某一被頻繁調用的功能的語句封裝,且不想通過函數方式
    的頭像 發表于 09-14 09:31 ?608次閱讀

    C語言函數怎樣實現封裝呢?

    函數,即包含多條語句的定義,其通常為某一被頻繁調用的功能的語句封裝,且不想通過函數方式
    的頭像 發表于 09-22 09:23 ?699次閱讀

    C語言函數的定義和用法

    函數是一特殊的函數,與普通函數不同的是,它可
    發表于 10-11 11:32 ?3330次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語言</b>中<b class='flag-5'>宏</b><b class='flag-5'>函數</b>的定義和用法

    c語言max函數在哪個庫

    C語言的max函數并不屬于標準C庫,它通常是由程序員自己編寫的一個定義。max函數用于比較兩個
    的頭像 發表于 11-22 10:18 ?3350次閱讀