C語言中的條件編譯


通常情況,我們想讓程序選擇性地執行,多會使用分支語句,比如if-else 或者switch-case 等。但有些時候,可能在程序的運行過程中,某個分支根本不會執行。

比如我們要寫一個跨平台項目,要求項目既能在Windows下運行,也能在Linux下運行。這個時候,如果我們使用if-else,如下:

Windows 有專有的宏_WIN32,Linux 有專有的宏__linux__

if(_WIN32)
    printf("Windows下執行的代碼\n");
else if(__linux__)
    printf("Linux下執行的代碼\n");
else
    printf("未知平台不能運行!\n");

 

 

這段代碼存在兩個問題:1、 在Windows下並沒有定義__linux__,編譯的時候會報錯,同樣在Linux中也沒有定義_WIN32。2、 假定這段程序可以運行,那么在Windows環境下另外兩個分支的代碼根本不可能運行,同理在Linux下也一樣。

 

處理這種情況我們可以使用條件編譯。條件編譯,顧名思義,就是根據一定的條件進行選擇性的編譯,我們要達到的效果,就是在Windows環境下另外兩個分支的語句根本不會編譯,這樣生成的可執行文件中,也不會還有對應語句的機器碼,這樣既提高了編譯效率,同時也減小了可執行文件的體積。

條件編譯通常可以用三種方式實現:

1、 #if--#elif--#else--#endif語句實現

通過這種方法實現的代碼為:

#if(_WIN32)
    printf("Windows下執行的代碼\n");
#elif (__linux__)
    printf("Linux下執行的代碼\n");
#else
    printf("未知平台不能運行!\n");
#endif

 

使用這種方式時需要注意,宏定義為真實#if才會執行,也就是說:

假如有宏定義#define _WIN32 0 這個時候#if是不會執行的。需要定義為#define _WIN32 1才會執行

2、 通過#ifdef--#else--#endif語句實現

通過這種方式實現的代碼為

#ifdef(_WIN32)
    printf("Windows下執行的代碼\n");
#else
    printf("Linux下執行的代碼\n");
#endif

這種方法下只需要定義了_WIN32就可以,沒有必要為真,也就是說

如果有宏定義#define _WIN32 0 上面#ifdef語句也是可以執行的,甚至#define _WIN32 上面的#ifdef也可以運行

 

當然也可以加入第一種方法中的#elif語句

#ifdef(_WIN32)
    printf("Windows下執行的代碼\n");
#elif (__linux__)
    printf("Linux下執行的代碼\n");
#else
    printf("未知平台不能運行!\n");
#end

但是需要注意的是,這種情況下,要想#elif語句執行__linux__的值必須為真!(同時沒有定義_WIN32)

 

3、 使用#ifndef語句,這種情況類似第二種,ifndef就是如果沒有定義宏,就執行。

 

在gcc編譯工具中

我們可以使用-D選項,動態地定義程序所需要的宏

比如我們可以這樣編譯 gcc test.c -o test -D _WIN32      這樣程序就可以在Windows下運行了(當然,實際情況是在Windows環境下,_WIN32已經被定義) gcc中的-D選項會默認將宏定義為1,如果要定義為其他的值使用等於號如:-D _WIN32=0       

 

很多的時候,尤其是實際的項目中,我們會使用cmake工具來構建自己的程序

 

在cmake中

我們可以在CMakeLists.txt中寫入ADD_DEIFNITIONS(-D _WIN32)來添加程序運行時用到的宏。但是這樣,一旦我們需要修改使用的宏,就要修改CMakeLists.txt文件,會很麻煩。

這時我們可以這樣做:

在CMakeLists.txt中寫入

IF(ENVIRO)
ADD_DEFINITIONS(-D _WIN32)
ENDIF(ENVIRO)

這樣,我們可以在使用cmake命令的時候加入-D選項,定義ENVIRO 命令如下

cmake -D ENVIRO=1,或者 cmake -D ENVIRO=ON

如果要取消這個定義可以使用: cmake -D ENVIRO=OFF 或 cmake -D ENVIRO=0 或者cmake -U ENVIRO

 

 

就寫到這里了,希望對你有幫助。。水平有限,有錯誤的地方還請諒解,並誠邀指正。。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM