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

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

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

c語(yǔ)言之條件編譯技巧與示例分析

開(kāi)源嵌入式 ? 來(lái)源:開(kāi)源嵌入式 ? 作者:開(kāi)源嵌入式 ? 2020-08-26 15:06 ? 次閱讀

C語(yǔ)言的條件編譯#if, #elif, #else, #endif、#ifdef, #ifndef

有些程序在調(diào)試、兼容性、平臺(tái)移植等情況下可能想要通過(guò)簡(jiǎn)單地設(shè)置一些參數(shù)就生成一個(gè)不同的軟件,這當(dāng)然可以通過(guò)變量設(shè)置,把所有可能用到的代碼都寫(xiě)進(jìn)去,在初始化時(shí)配置,但在不同的情況下可能只用到一部分代碼,就沒(méi)必要把所有的代碼都寫(xiě)進(jìn)去,就可以用條件編譯,通過(guò)預(yù)編譯指令設(shè)置編譯條件,在不同的需要時(shí)編譯不同的代碼。

(一)條件編譯方法

條件編譯是通過(guò)預(yù)編譯指令來(lái)實(shí)現(xiàn)的,主要方法有:

1、#if, #elif, #else, #endif

#if條件1

代碼段1

#elif 條件2

代碼段2

。..

#elif 條件n

代碼段n

#else

代碼段 n+1

#endif

即可以設(shè)置不同的條件,在編譯時(shí)編譯不同的代碼,預(yù)編譯指令中的表達(dá)式與C語(yǔ)言本身的表達(dá)式基本一至如邏輯運(yùn)算、算術(shù)運(yùn)算、位運(yùn)算等均可以在預(yù)編譯指令中使用。之所以能夠?qū)崿F(xiàn)條件編譯是因?yàn)轭A(yù)編譯指令是在編譯之前進(jìn)行處理的,通過(guò)預(yù)編譯進(jìn)行宏替換、條件選擇代碼段,然后生成最后的待編譯代碼,最后進(jìn)行編譯。

#if的一般含義是,如果#if后面的常量表達(dá)式為true,則編譯它所控制的代碼,如條件1成立時(shí)就代碼段1,條件1不成立再看條件2是否成立,如果條件2成立則編譯代碼段2,否則再依次類(lèi)推判斷其它條件,如果條件1-N都不成力則會(huì)編譯最后的代碼段n+1.

2、#ifdef, #else, #endif或#ifndef, #else, #endif

條件編譯的另一種方法是用#ifdef與#ifndef命令,它們分別表示“如果有定義”及“如果無(wú)定義”。有定義是指在編譯此段代碼時(shí)是否有某個(gè)宏通過(guò) #define 指令定義的宏,#ifndef指令指找不到通過(guò)#define定義的某宏,該宏可以是在當(dāng)前文件此條指令的關(guān)面定義的,也可以是在其它文件中,但在此指令之前包含到該文件中的。

#ifdef的一般形式是:

#ifdef macro_name

代碼段1

#else

代碼段2

#endif

#ifdef的一般形式是:

#ifndef macro_name

代碼段2

#else

代碼段1

#endif

這兩段代碼的效果是完全一樣的。

3、通過(guò)宏函數(shù)defined(macro_name)

參數(shù)為宏名(無(wú)需加“”),如果該macro_name定義過(guò)則返回真,否則返回假,用該函數(shù)則可以寫(xiě)比較復(fù)雜的條件編譯指令如

#if defined(macro1) || (!defined(macro2) && defined(macro3))

。..

#else

。..

#endif

(二)條件編譯技巧與示例

(1)#ifdef和#defined()比較

首先比較一下這兩種方法,第一種方法只能判斷一個(gè)宏,如果條件比較復(fù)雜實(shí)現(xiàn)起來(lái)比較煩鎖,用后者就比較方便。如有兩個(gè)宏MACRO_1,MACRO_2只有兩個(gè)宏都定義過(guò)才會(huì)編譯代碼段A,分別實(shí)現(xiàn)如下:

#ifdef MACRO_1

#ifdef MACRO_2

代碼段A

#endif

#endif

或者

#if defined(MACRO_1) && defined(MACRO_2)

#endif

同樣,要實(shí)現(xiàn)更復(fù)雜的條件用#ifdef更麻煩,所以推薦使用后者,因?yàn)榧词巩?dāng)前代碼用的是簡(jiǎn)單的條件編譯,以后在維護(hù)、升級(jí)時(shí)可能會(huì)增加,用后者可維護(hù)性較強(qiáng)。舊的編譯器可能沒(méi)有實(shí)現(xiàn)#defined()指令,C99已經(jīng)加為標(biāo)準(zhǔn)。要兼容老的編譯器,還需用#ifdef指令。

2、#if與 #ifdef或#ifdefined()比較

比如自己寫(xiě)了一個(gè)printf函數(shù),想通過(guò)一個(gè)宏MY_PRINTF_EN實(shí)現(xiàn)條件編譯,用#if可實(shí)現(xiàn)如下

C語(yǔ)言的條件編譯。

#define MY_PRINTF_EN 1

#if MYS_PRINTF_EN == 1

int printf(char* fmt, char* args, 。..)

{

。..

}

#endif

如果宏MY_PRINTF_EN定義為1則編譯這段代碼,如果宏定義不為1或者沒(méi)有定義該宏,則不編譯這段代碼。同樣也可以通過(guò)#ifdef或者#defined()實(shí)現(xiàn),如

#define MY_PRINTF_EN 1

#if defined(MY_PRINTF_EN)

int printf(char* fmt, char* args, 。..)

{

。..

}

#endif

在這種情況下兩種方法具有異曲同工之妙,但試想如果你為了節(jié)約代碼寫(xiě)了兩個(gè)printf函數(shù),在不同情況下使用不同的printf函數(shù),一個(gè)是精簡(jiǎn)版一個(gè)是全功能標(biāo)準(zhǔn)版,如:

#define MY_PRINTF_SIMPLE

#ifdef MY_PRINTF_SIMPLE

void printf(*str) // 向終端簡(jiǎn)單地輸出一個(gè)字符串

{。..

}

#endif

#ifdef MY_PRINTF_STANDARD

int printf(char* fmt, char* args, 。..)

{。..

}

#endif

同樣可以用#if defined()實(shí)現(xiàn)

#define MY_PRINTF_SIMPLE

#if defined(MY_PRINTF_SIMPLE)

void printf(*str) // 向終端簡(jiǎn)單地輸出一個(gè)字符串

{

。..

}

#elif defined(MY_PRINTF_STANDARD)

int printf(char* fmt, char* args, 。..)

{

。..

}

#endif

兩種方法都可以實(shí)現(xiàn),但可見(jiàn)后者更方便。但試想如果你有三個(gè)版本,用前者就更麻煩了,但方法相似,用后者就更方便,但仍需三個(gè)宏進(jìn)行控制,你要住三個(gè)宏,改進(jìn)一下就用#if可以用一個(gè)宏直接控制N種情況如:

#define MY_PRINTF_VERSION 1

#if MY_PRINTF_VERSION == 1

void printf(*str) // 向終端簡(jiǎn)單地輸出一個(gè)字符串

{

。..

}

#elif MY_PRINTF_VERSION == 2

int printf(char* fmt, char* args, 。..)

{

。..

}

#elif MY_PRINTF_VERSION == 3

int printf(unsigned char com_number, char* str)

{

。..

}

#else

默認(rèn)版本

#endif

這樣,你只需修改一下數(shù)字就可以完成版本的選擇了

看來(lái)好像用#if 比較好了,試想如下情況:你寫(xiě)了一個(gè)配置文件叫做config.h用來(lái)配置一些宏,通過(guò)這些宏來(lái)控制代碼,如你在config.h的宏

#define MY_PRINTF_EN 1

來(lái)控制是否需要編譯自己的printf函數(shù),而在你的源代碼文件printf.c中有如下指令

#i nclude “config.h”

#if MY_PRINTF_EN == 1

int printf(char* fmt, char* args, 。..)

{

。..

}

#endif

但這樣也會(huì)有一個(gè)問(wèn)題,就是如果你忘了在config.h中添加宏MY_PRINTF_EN,那么自己寫(xiě)的printf函數(shù)也不會(huì)被編譯,有些編譯器會(huì)給出警告:MY_PRINTF_EN未定義。如果你有兩個(gè)版本的想有一個(gè)默認(rèn)版本,可以在printf.c中這樣實(shí)現(xiàn)

#incldue “config.h”

#if !defined(MY_PRINTF_VERSION)

#define MY_PRINTF_VERSION 1

#endif

#if MY_PRINTF_VERSION == 1

void printf(*str) // 向終端簡(jiǎn)單地輸出一個(gè)字符串

{

。..

}

#elif MY_PRINTF_VERSION == 2

int printf(char* fmt, char* args, 。..)

{

。..

}

#elif MY_PRINTF_VERSION == 3

int printf(unsigned char com_number, char* str)

{

。..

}

#endif

這種情況下還得用到#ifdef或#ifdefined(),你可以不用動(dòng)主體的任何代碼,只需要修改printf.c文件中MY_RPINTF_VERSION宏的數(shù)字就可以改變了,如果用前面那種方法還得拖動(dòng)代碼,在拖動(dòng)中就有可能造成錯(cuò)誤。

再試想,如果軟件升級(jí)了,或者有了大的改動(dòng),原來(lái)有三個(gè)版本,現(xiàn)在只剩下兩個(gè)版本了,如

#if MY_PRINTF_VERSION == 2

int printf(char* fmt, char* args, 。..)

{

。..

}

#elif MY_PRINTF_VERSION == 3

int printf(unsigned char com_number, char* str)

{

。..

}

#endif

因?yàn)檫@些核心代碼不想讓使用這些代碼的人關(guān)心,他們只需要修改config.h文件,那就要在printf.c中實(shí)現(xiàn)兼容性。如果以前有人在config.h配置宏MY_PRINTF_VERSION為1,即有

#define MY_PRINTF_VERSION 1

而現(xiàn)在沒(méi)有1版本了,要想兼容怎么辦?那當(dāng)然可以用更復(fù)雜的條件實(shí)現(xiàn)如:

#if MY_PRINTF_VERSION == 2 || MY_PRINTF_VERSION == 1

int printf(char* fmt, char* args, 。..)

{

。..

}

#elif MY_PRINTF_VERSION == 3

int printf(unsigned char com_number, char* str)

{

。..

}

#endif

不過(guò)還有另外一種方法,即使用#undef命令

#if MY_PRINTF_VERSION == 1

#undef MY_PRINTF_VERSION

#define MY_PRINTF_VERSION 2

#endif

#if MY_PRINTF_VERSION == 2

int printf(char* fmt, char* args, 。..)

{

。..

}

#elif MY_PRINTF_VERSION == 3

int printf(unsigned char com_number, char* str)

{

。..

}

#endif

用#if還有一個(gè)好處,如果你把宏名記錯(cuò)了,把MY_PRINTF_EN定義成了MY_PRINT_EN,那么你用#ifdef MY_PRINTF_EN或者#if defined(MY_PRINTF_EN)控制的代碼就不能被編譯,查起來(lái)又不好查,用#if MY_PRINTF_EN ==1控制就很好查,因?yàn)槟惆袽Y_PRINTF_EN定義成MY_PRINT_EN,則MY_PRINTF_EN實(shí)際上沒(méi)有定義,那么編譯器會(huì)給出警告#if MY_PRINTF_EN== 1中的MY_PRINTF_EN沒(méi)有定義,但錯(cuò)就比較快。

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

    關(guān)注

    180

    文章

    7575

    瀏覽量

    134131
  • 條件編譯
    +關(guān)注

    關(guān)注

    0

    文章

    3

    瀏覽量

    5488

原文標(biāo)題:【單片機(jī)編程技術(shù)】單片機(jī)基礎(chǔ)c語(yǔ)言之條件編譯詳解

文章出處:【微信號(hào):KY_QRS,微信公眾號(hào):開(kāi)源嵌入式】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    幾款C語(yǔ)言編譯器推薦

    一些剛開(kāi)始接觸C語(yǔ)言編譯的網(wǎng)友想下載一款C語(yǔ)言編譯器來(lái)使用,不過(guò),網(wǎng)絡(luò)上有不少
    發(fā)表于 09-05 09:19 ?1w次閱讀

    C語(yǔ)言編譯過(guò)程

    C語(yǔ)言編譯鏈接過(guò)程要把我們編寫(xiě)的一個(gè)C程序源代碼,轉(zhuǎn)換成可以在硬件上運(yùn)行的程序(可執(zhí)行代碼),需要進(jìn)行編譯和鏈接。過(guò)程圖解如下。
    發(fā)表于 06-25 10:36 ?352次閱讀
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>的<b class='flag-5'>編譯</b>過(guò)程

    C語(yǔ)言之正確使用Const

    C語(yǔ)言之正確使用Const基本解釋  const是一個(gè)C語(yǔ)言的關(guān)鍵字,它限定一個(gè)變量不允許被改變。使用const在一定程度上可以提高程序的健壯性,另外,在觀看別人代碼的時(shí)候,清晰理解c
    發(fā)表于 10-07 11:20

    c語(yǔ)言之高手進(jìn)階

    c語(yǔ)言之高手進(jìn)階 從點(diǎn)滴開(kāi)始 楊帆起航
    發(fā)表于 07-04 16:14

    C語(yǔ)言條件編譯的妙用

    線。我對(duì)學(xué)習(xí)總結(jié)出一句話“壓力不是有人努力,而是比你牛X N倍的人依然在努力” C語(yǔ)言條件編譯的應(yīng)用 一般情況下,源程序中所有的行都要參加編譯
    發(fā)表于 07-19 01:53

    C語(yǔ)言編譯

    電子發(fā)燒友網(wǎng)站提供《C語(yǔ)言編譯器.exe》資料免費(fèi)下載
    發(fā)表于 01-15 17:45 ?50次下載

    C語(yǔ)言之詳解_ifdef等宏及妙用

    C語(yǔ)言之詳解_ifdef等宏及妙用的教程
    發(fā)表于 11-16 19:03 ?0次下載

    C語(yǔ)言之自然對(duì)數(shù)的底e的計(jì)算

    C語(yǔ)言之自然對(duì)數(shù)的底e的計(jì)算,很好的C語(yǔ)言資料,快來(lái)學(xué)習(xí)吧。
    發(fā)表于 04-22 17:45 ?0次下載

    淺談C、C++ 和 ARM 匯編語(yǔ)言之間的調(diào)用

    12.4 C target=_blank style=cursor:pointer;color:#D05C38;text-decoration:underline;》C、C++ 和 A
    發(fā)表于 10-19 09:24 ?2次下載

    單片機(jī)C語(yǔ)言之串口通信協(xié)議

    本文首先介紹了串口的基本結(jié)構(gòu),其次介紹了串口通信方式,最后詳細(xì)介紹了單片機(jī)C語(yǔ)言之串口通信協(xié)議。
    發(fā)表于 05-22 12:37 ?4779次閱讀
    單片機(jī)<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言之</b>串口通信協(xié)議

    解析C語(yǔ)言編譯過(guò)程中所做的工作

    C語(yǔ)言編譯鏈接過(guò)程要把我們編寫(xiě)的一個(gè)C程序源代碼,轉(zhuǎn)換成可以在硬件上運(yùn)行的程序(可執(zhí)行代碼),需要進(jìn)行編譯和鏈接。過(guò)程圖解如下: 本文講解
    的頭像 發(fā)表于 06-27 10:21 ?3043次閱讀
    解析<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b><b class='flag-5'>編譯</b>過(guò)程中所做的工作

    C語(yǔ)言條件編譯語(yǔ)句and單片機(jī)DMA的介紹

    C語(yǔ)言條件編譯語(yǔ)句and單片機(jī)DMA的介紹C語(yǔ)言條件
    發(fā)表于 11-29 10:36 ?3次下載
    <b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b><b class='flag-5'>條件</b><b class='flag-5'>編譯</b>語(yǔ)句and單片機(jī)DMA的介紹

    C語(yǔ)言條件編譯詳解

    —般情況下,C語(yǔ)言源程序中的每一行代碼.都要參加編譯。但有時(shí)候出于對(duì)程序代碼優(yōu)化的考慮,希望只對(duì)其中一部分內(nèi)容進(jìn)行編譯。此時(shí)就需要在程序中加上條件
    的頭像 發(fā)表于 04-14 13:12 ?2903次閱讀

    C語(yǔ)言之父你知道是誰(shuí)嗎

    隨著C語(yǔ)言在多個(gè)領(lǐng)域的推廣、應(yīng)用,一些新的特性不斷被各種編譯器實(shí)現(xiàn)并添加進(jìn)來(lái)。于是,建立一個(gè)新的“無(wú)歧義、與具體平臺(tái)無(wú)關(guān)的 C 語(yǔ)言定義”
    發(fā)表于 03-16 10:49 ?3515次閱讀

    為什么C語(yǔ)言要進(jìn)行編譯

    為什么我們編寫(xiě)的C語(yǔ)言要進(jìn)行編譯?什么是編譯編譯時(shí)發(fā)生了什么? 機(jī)器無(wú)法理解我們編寫(xiě)的C
    的頭像 發(fā)表于 11-24 15:47 ?953次閱讀
    為什么<b class='flag-5'>C</b><b class='flag-5'>語(yǔ)言</b>要進(jìn)行<b class='flag-5'>編譯</b>