C語言中條件編譯詳解。本站提示廣大學習愛好者:(C語言中條件編譯詳解)文章只能為提供參考,不一定能成為您想要的結果。以下是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
就寫到這裡了,希望對你有幫助。。水平有限,有錯誤的地方還請諒解,並誠邀指正。。