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

有些程序在調試、兼容性、平臺移植等情況下可能想要通過簡單地設置一些參數就生成一個不同的軟件,這當然可以通過變量設置,把所有可能用到的代碼都寫進去,在初始化時配置,但在不同的情況下可能只用到一部分代碼,就沒必要把所有的代碼都寫進去,就可以用條件編譯,通過預編譯指令設置編譯條件,在不同的需要時編譯不同的代碼

?

   (一)條件編譯方法

   條件編譯是通過預編譯指令來實現的,主要方法有:

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

 #if 條件 1
 代碼段 1
#elif 條件 2
   代碼段 2
...
#elif 條件 n
 代碼段 n
#else
 代碼段 n+1
#endif

  即可以設置不同的條件,在編譯時編譯不同的代碼,預編譯指令中的表達式與C語言本身的表達式基本一至如邏輯運算、算術運算、位運算等均可以在預編譯指令中使用。之所以能夠實現條件編譯是因為預編譯指令是在編譯之前進行處理的,通過預編譯進行宏替換、條件選擇代碼段,然后生成最后的待編譯代碼,最后進行編譯。

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

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

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

#ifdef的一般形式是:

 #ifdef macro_name
    代碼段 1
#else
    代碼段 2
#endif


#ifdef的一般形式是:

#ifndef macro_name
    代碼段 2
#else
    代碼段 1
#endif

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

   3、通過宏函數defined(macro_name)

  參數為宏名(無需加""),如果該macro_name定義過則返回真,否則返回假,用該函數則可以寫比較復雜的條件編譯指令如

 #if defined(macro1) || (!defined(macro2) && defined(macro3))
...
#else
...
#endif

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

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

  首先比較一下這兩種方法,第一種方法只能判斷一個宏,如果條件比較復雜實現起來比較煩鎖,用后者就比較方便。如有兩個宏MACRO_1,MACRO_2只有兩個宏都定義過才會編譯代碼段A,分別實現如下:

 #ifdef MACRO_1
#ifdef MACRO_2
    代碼段 A
#endif
#endif

或者
#if defined(MACRO_1) && defined(MACRO_2)
#endif

  同樣,要實現更復雜的條件用#ifdef更麻煩,所以推薦使用后者,因為即使當前代碼用的是簡單的條件編譯,以后在維護、升級時可能會增加,用后者可維護性較強。舊的編譯器可能沒有實現#defined()指令,C99已經加為標準。要兼容老的編譯器,還需用#ifdef指令。

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

   比如自己寫了一個printf函數,想通過一個宏MY_PRINTF_EN實現條件編譯,用#if可實現如下

C語言的條件編譯。

?

 #define MY_PRINTF_EN 1

#if MYS_PRINTF_EN == 1
 int printf(char* fmt, char* args, ...)
{
    ...
}
#endif

  如果宏MY_PRINTF_EN定義為1則編譯這段代碼,如果宏定義不為1或者沒有定義該宏,則不編譯這段代碼。同樣也可以通過#ifdef或者#defined()實現,如

 #define MY_PRINTF_EN 1

#if defined(MY_PRINTF_EN)
 int printf(char* fmt, char* args, ...)
{
    ...
}
#endif

  在這種情況下兩種方法具有異曲同工之妙,但試想如果你為了節約代碼寫了兩個printf函數,在不同情況下使用不同的printf函數,一個是精簡版一個是全功能標準版,如:

 #define MY_PRINTF_SIMPLE

#ifdef MY_PRINTF_SIMPLE
   void printf(*str) // 向終端簡單地輸出一個字符串
{...
}
#endif
#ifdef MY_PRINTF_STANDARD
 int printf(char* fmt, char* args, ...)
{...
}
#endif

同樣可以用#if defined()實現
#define MY_PRINTF_SIMPLE

#if defined(MY_PRINTF_SIMPLE)
   void printf(*str) // 向終端簡單地輸出一個字符串
{
    ...
}
#elif defined(MY_PRINTF_STANDARD)
 int printf(char* fmt, char* args, ...)
{
    ...
}
#endif

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

 #define MY_PRINTF_VERSION     1

#if MY_PRINTF_VERSION == 1
   void printf(*str) // 向終端簡單地輸出一個字符串
{
    ...
}
#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
    默認版本
#endif

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

   看來好像用#if 比較好了,試想如下情況:你寫了一個配置文件叫做config.h用來配置一些宏,通過這些宏來控制代碼,如你在config.h的宏

   #define MY_PRINTF_EN 1

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

 #i nclude "config.h"
#if MY_PRINTF_EN == 1
 int printf(char* fmt, char* args, ...)
{
    ...
}
#endif

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

 #incldue "config.h"
#if !defined(MY_PRINTF_VERSION)
  #define MY_PRINTF_VERSION   1
#endif

#if MY_PRINTF_VERSION == 1
   void printf(*str) // 向終端簡單地輸出一個字符串
{
    ...
}
#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或#if defined(),你可以不用動主體的任何代碼,只需要修改printf.c文件中MY_RPINTF_VERSION宏的數字就可以改變了,如果用前面那種方法還得拖動代碼,在拖動中就有可能造成錯誤。

   再試想,如果軟件升級了,或者有了大的改動,原來有三個版本,現在只剩下兩個版本了,如

 #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

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

   #define MY_PRINTF_VERSION   1

   而現在沒有1版本了,要想兼容怎么辦?那當然可以用更復雜的條件實現如:

 #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

   不過還有另外一種方法,即使用#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還有一個好處,如果你把宏名記錯了,把MY_PRINTF_EN定義成了MY_PRINT_EN,那么你用#ifdef MY_PRINTF_EN或者#if defined(MY_PRINTF_EN)控制的代碼就不能被編譯,查起來又不好查,用#if MY_PRINTF_EN ==1控制就很好查,因為你把MY_PRINTF_EN定義成MY_PRINT_EN,則MY_PRINTF_EN實際上沒有定義,那么編譯器會給出警告#if MY_PRINTF_EN == 1中的MY_PRINTF_EN沒有定義,但錯就比較快。

              ???????????????????????????????????????????????????? 轉至博主Embeder

轉載于:https://www.cnblogs.com/droidxin/p/3446451.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/376261.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/376261.shtml
英文地址,請注明出處:http://en.pswp.cn/news/376261.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

山體等高線怎么看_每日一題 | 此處向斜山,你看出來了嗎?

每日一題 | 此處向斜山,你看出來了嗎?(2018江蘇高考)如圖為某區域地質簡圖。該區沉積地層有Q、P、C、D、S2、S1,其年代依次變老。讀圖回答1~2題。1.從甲地到乙地的地形地質剖面示意圖是(  )2.為揭示深部地…

cmake The source directory xxxx does not appear to contain CMakeLists.txt

執行 cmake . 的時候報錯: The source directory “xxxx” does not appear to contain CMakeLists.txt 簡單來說就是當前文件夾里面沒有 CMakeLists.txt

SSH出錯--hibernate--org.hibernate.hql.ast.QuerySyntaxException: User is not mapped [from User]

String queryString "from user where u.userName ? and u.userPassword ?"; ----------------------------------------------------------- 改為: String queryString "from User where u.userName ? and u.userPassword ?"; 我估…

Linux下的tar壓縮解壓縮命令詳解

tar -c: 建立壓縮檔案-x:解壓-t:查看內容-r:向壓縮歸檔文件末尾追加文件-u:更新原壓縮包中的文件 這五個是獨立的命令,壓縮解壓都要用到其中一個,可以和別的命令連用但只能用其中一個。下面的參數是根據需要…

java和c++的區別大嗎_大空間消防水炮ZDMS0.8/30S坐裝和吊裝有區別嗎?

大空間消防水炮現在是高大建筑的消防必備的設備之一,其型號按照流量可分為4種,ZDMS0.6/5S,ZDMS0.6/10S,SZDMS0.8/20S,ZDMS0.8/30S。在這中間使用較多的是5L和30L的,5L的消防水炮都是吊裝,但是30…

Windows Hook(1)加載DLL

DLL代碼 #include <Windows.h> BOOL APIENTRY DllMain( HMODULE hModule,DWORD ul_reason_for_call,LPVOID lpReserved) {switch (ul_reason_for_call){case DLL_PROCESS_ATTACH:MessageBox(NULL, L"dllHook", L"Hook", MB_OK);break;case DLL_THR…

WPF Delegate委托整理

那啥&#xff0c;是從這里整理出來的&#xff0c;感謝Rising_Sun&#xff0c;整理的過于簡單&#xff0c;看不明白的戳這里 using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Windows; using System.Windows.Controls; us…

silverligh的數據訪問

對于在Silverlight中訪問數據&#xff0c;初學者的誤解之一就是他們在Silverlight中尋找ADO.NET類庫。別找了&#xff0c;找不到的。記住&#xff0c;Silverlight是部署在互聯網上的客端技術&#xff0c;你不能要求一個瀏覽器插件去直接訪問你的數據庫……除非你想把數據庫直接…

cacheinterceptor第二次訪問沒被調用_訪問者設計模式在OSG中的應用

為什么要談談訪問者設計模式呢&#xff1f;因為OSG整個引擎就是用訪問者設計模式建立起來的&#xff0c;不論是遍歷節點圖&#xff0c;還是做各種實用的功能&#xff0c;都需要大量的用到訪問者設計模式。先談談訪問者設計模式的定義。1&#xff1a;什么是訪問者模式訪問者模式…

Windows Hook(2)調用DLL函數

DLL代碼 #include <Windows.h>BOOL APIENTRY DllMain( HMODULE hModule,DWORD ul_reason_for_call,LPVOID lpReserved) {switch (ul_reason_for_call){case DLL_PROCESS_ATTACH:MessageBox(NULL, L"dllHook", L"Hook", MB_OK);break;case DLL_THRE…

HDU4678_Mine

很有意思&#xff0c;很好的題目。 這樣的&#xff0c;一個n*m的掃雷地圖&#xff0c;告訴你哪些地方是有雷的。一個人如果點在了空白處&#xff0c;那么與其相鄰的&#xff08;八個方向&#xff09;的數字以及空白都會遞歸地顯示出來&#xff0c;如果點在數字上面&#xff0c;…

pygame只能編寫游戲_游戲框架搭建

游戲框架搭建目標 —— 使用 面相對象 設計 飛機大戰游戲類目標明確主程序職責實現主程序類準備游戲精靈組01. 明確主程序職責回顧 快速入門案例&#xff0c;一個游戲主程序的 職責 可以分為兩個部分&#xff1a;游戲初始化游戲循環根據明確的職責&#xff0c;設計 PlaneGame 類…

周末閱讀:本周熱門文章排行榜

那道不清說不盡的故事 iPhone 的創意并非來自喬布斯一人&#xff0c;其起源可以追溯到 Jony 的設計團隊對多點觸控屏幕的思考和探索&#xff0c;也正是因為對這個技術的看好&#xff0c;在對其在手機上的可行新的不斷測試后&#xff0c;蘋果最后下定決心進軍手機領域。這篇文章…

python3 hash算法使用

python3下的pycryptodome庫 from Crypto.cipher import * if __name__ __main__:message 123#MD5和SHA的用法差不多print("SHA3_512: " SHA3_512.new(message.encode(utf-8)).digest().hex())print("SHA512: " SHA512.new(message.encode(utf-8)).dig…

poj3335 半平面交

題意&#xff1a;給出一多邊形。判斷多邊形是否存在一點&#xff0c;使得多邊形邊界上的所有點都能看見該點。 sol&#xff1a;在紙上隨手畫畫就可以找出規律&#xff1a;按逆時針順序連接所有點。然后找出這些line的半平面交。 題中給出的點已經按順時針排好序了&#xff0c;所…

php進程間通信 yoc_續上篇Swoole多進程數據共享的問題

原因進程作為程序執行過程中資源分配的基本單位&#xff0c;擁有獨立的地址空間,同一進程的線程可以共享本進程的全局變量&#xff0c;靜態變量等數據和地址空間&#xff0c;但進程之間資源相互獨立。由于PHP語言不支持多線程&#xff0c;因此Swoole使用多進程模式&#xff0c;…

JavaBean的規范

&#xff08;1&#xff09;JavaBean 類必須是一個公共類&#xff0c;并將其訪問屬性設置為 public &#xff08;2&#xff09;JavaBean 類必須有一個空的構造函數&#xff1a;類中必須有一個不帶參數的公用構造器&#xff0c;此構造器也應該通過調用各個特性的設置方法來設置特…

linux虛擬機ip修改無效

把一個centos虛擬機移動到另一臺電腦的時候&#xff0c;移動前是靜態ip&#xff0c;移動后發現虛擬機的ip不同了。 由于使用的是NAT&#xff0c;于是就修改了虛擬機的配置&#xff0c;發現虛擬機的ip仍然不是配置文件需要的情況。 可以嘗試命令nmcli con show&#xff0c;如果…

驗證(Verification)與確認(Validation)的差別

驗證(Verification)與確認&#xff08;Validation&#xff09;的差別 說法一&#xff1a; &#xff08;2&#xff09;“驗證(Verification)”的涵義 通過提供客觀證據對規定要求已得到滿足的認定。 &#xff08;2&#xff09;“確認&#xff08;Validation&#xff09;”的涵義…

vscode自動格式化不符合eslint_VsCode(Visual Studio Code)格式化代碼符合EsLint

利用Visual Studio Code ESlint插件&#xff0c;實現自動格式化代碼步驟一&#xff1a;安裝ESlint插件>點擊Extensions或者CtrlShiftX>搜索ESlint>install EsLint步驟二: 重啟VsCode&#xff0c; 發現代碼提示報錯&#xff0c;代碼不符合規范步驟三&#xff1a;鼠標ho…