預處理指令 preprocessor directives
define
undef
ifdef ifndef if endif else and elif
line
error
include
預定義識別符號
pragma
預處理指令是我們寫在程式**中的給預處理器(preprocessor)的命令,而不是程式本身的語句。預處理器在我們編譯乙個c++程式時由編譯器自動執行,它負責控制對程式**的第一次驗證和消化。
所有這些指令必須寫在單獨的一行中,它們不需要加結尾的(;)分號。
在這個教程的開頭我們已經提到了一種預處理指令: #define ,可以被用來生成巨集定義常量(defined constantants 或 macros),它的形式是:
#define name value
它的作用是定義乙個叫做name 的巨集定義,然後每當在程式中遇到這個名字的時候,它就會被value代替,例如:
#define max_width 100
char str1[max_width];
char str2[max_width];
它定義了兩個最多可以儲存100個字元的字串。
#define 也可以被用來定義巨集函式:
#define getmax(a,b) ((a)>(b)?(a):(b))
int x=5, y;
y = getmax(x,2);
這段**執行後y 的值為5 。
#undef 完成與 #define相反的工作,它取消對傳入的引數的巨集定義:
#define max_width 100
char str1[max_width];
#undef max_width
如果在這段**之後,使用max_width編譯器就會報錯。
這些指令可以使程式的一部分在某種條件下被忽略。
#ifdef 可以使一段程式只有在某個指定常量已經被定義了的情況下才被編譯,無論被定義的值是什麼。它的操作是:
#ifdef name
// code here
#endif
例如:#ifdef max_width
char str[max_width];
#endif
在這個例子中,語句char str[max_width]; 只有在巨集定義常量max_width 已經被定義的情況下才被編譯器考慮,不管它的值是什麼。如果它還沒有被定義,這一行**則不會被包括在程式中。
#ifndef 起相反的作用:在指令#ifndef 和 #endif 之間的**只有在某個常量沒有被定義的情況下才被編譯,例如:
#ifndef max_width
#define max_width 100
#endif
char str[max_width];
這個例子中,如果當處理到這段**的時候max_width 還沒有被定義,則它會被定義為值100。而如果它已經被定義了,那麼它會保持原值 (因為#define 語句這一行不會被執行) 。
指令#if, #else 和 #elif (elif = else if) 用來使得其後面所跟的程式部分只有在特定條件下才被編譯。這些條件只能夠是常量表示式,例如:
const int value_judge = 150;
#if value_judge >200
#define max_width 200
#elif value_judge <50
#define max_width 50
#else
#define max_width 100
#endif
char str[max_width];
注意看這一連串的指令 #if, #elif 和 #else 是怎樣以 #endif 結尾的。
當我們編譯一段程式的時候,如果有錯誤發生,編譯器會在錯誤前面顯示出錯檔案的名稱以及檔案中的第幾行發生的錯誤。
指令#line 可以使我們對這兩點進行控制,也就是說當出錯時顯示檔案中的行數以及我們希望顯示的檔名。它的格式是:
#line number "filename"
這裡number 是將會賦給下一行的新行數。它後面的行數從這一點逐個遞增。
filename 是乙個可選引數,用來替換自此行以後出錯時顯示的檔名,直到有另外乙個#line指令替換它或直到檔案的末尾。例如:
#line 1 "assigning variable"
int a?;
這段**將會產生乙個錯誤,顯示為在檔案"assigning variable", line 1 。
這個指令將中斷編譯過程並返回乙個引數中定義的出錯資訊,例如:
#ifndef __cplusplus
#error a c++ compiler is required
#endif
這個例子中如果 __cplusplus 沒有被定義就會中斷編譯過程。
這個指令我們已經見到很多次。當預處理器找到乙個#include 指令時,它用指定檔案的全部內容替換這條語句。宣告包含乙個檔案有兩種方式:
#include "file"
#include
兩種表達的唯一區別是編譯器應該在什麼路經下尋找指定的檔案。第一種情況下,檔名被寫在雙引號中,編譯器首先在包含這條指令的檔案所在的目錄下進行尋找,如果找不到指定檔案,編譯器再到被配置的預設路徑下(也就是標準標頭檔案路徑下)進行尋找。
如果檔名是在尖括號 <> 中,編譯器會直接到預設標準標頭檔案路徑下尋找。
預定義識別符號
為了處理一些有用的資訊,預處理定義了一些預處理識別符號,雖然各種編譯器的預處理識別符號不盡相同,但是他們都會處理下面的4種:
__file__ 正在編譯的檔案的名字
__line__ 正在編譯的檔案的行號
__date__ 編譯時刻的日期字串,例如: "25 dec 2000"
__time__ 編譯時刻的時間字串,例如: "12:30:55"
例如:cout<<"the file is :"<<__file__"<<"! the lines is:"<<__line__<
C C 預處理指令
預處理指令是以 號開頭的 行。號必須是該行除了任何空白字元外的第乙個字元。後是指令關鍵字,在關鍵字和 號之間允許存在任意個數的空白字元。整行語句構成了一條預處理指令,該指令將在編譯器進行編譯之前對源 做某些轉換。1.和 1 define pi 3.1415926 2 define paste n a...
C C 編譯預處理指令
眾所周知,你的程式編譯前要做的事就是掃瞄源 對其做初步的轉換,產生新的源 提供給編譯器,這個過程就叫編譯預處理。這個處理過程由預處理器來完成,預處理器是在程式真正執行前由編譯器呼叫的預處理程式。常見的預處理有以下三種 include 是一種最為常見的預處理,主要是做為檔案的引用組合源程式正文。巨集替...
C C 常用預處理指令
預處理是在編譯之前的處理,而編譯工作的任務之一就是語法檢查,預處理不做語法檢查。預處理命令以符號 開頭。常用的預處理指令包括 巨集定義 巨集定義又稱為巨集代換 巨集替換,簡稱 巨集 巨集替換只作替換,不做計算,不做表示式求解。巨集定義分帶引數的巨集定義和不帶引數的巨集定義。在帶引數的巨集定義,巨集名...