預處理語句是由一系列和預處理相關的命令符組成的.預處理語句以#作為起始標記,其後緊跟預處理命令關鍵字,之後是空格,空格之後是預處理命令的內容.C++提供多種預處理功能,如巨集定義,文件包括,條件編譯等. #define 在這個教程的開頭我們已經提到了一種預處理指令: #define ,可以被用來生成巨集 ...
預處理語句是由一系列和預處理相關的命令符組成的.預處理語句以#作為起始標記,其後緊跟預處理命令關鍵字,之後是空格,空格之後是預處理命令的內容.C++提供多種預處理功能,如巨集定義,文件包括,條件編譯等.
#define
在這個教程的開頭我們已經提到了一種預處理指令: #define ,可以被用來生成巨集定義常量(defined constantants 或 macros),它的形式是:
#define name value它的作用是定義一個叫做name 的巨集定義,然後每當在程式中遇到這個名字的時候,它就會被value代替,例如:
#define MAX_WIDTH 100char str1[MAX_WIDTH];
char str2[MAX_WIDTH];
它定義了兩個最多可以存儲100個字元的字元串。
#define 也可以被用來定義巨集函數:
#define getmax(a,b) a>b?a:bint x=5, y;
y = getmax(x,2);
這段代碼執行後y 的值為5 。
#undef
#undef 完成與 #define相反的工作,它取消對傳入的參數的巨集定義:
#define MAX_WIDTH 100char str1[MAX_WIDTH];
#undef MAX_WIDTH
#define MAX_WIDTH 200
char str2[MAX_WIDTH];
#ifdef, #ifndef, #if, #endif, #else and #elif
這些指令可以使程式的一部分在某種條件下被忽略。
#ifdef 可以使一段程式只有在某個指定常量已經被定義了的情況下才被編譯,無論被定義的值是什麼。它的操作是:
#ifdef name// code here
#endif
例如:
#ifdef MAX_WIDTHchar 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) 用來使得其後面所跟的程式部分只有在特定條件下才被編譯。這些條件只能夠是常量表達式,例如:
#if MAX_WIDTH>200#undef MAX_WIDTH
#define MAX_WIDTH 200
#elsif MAX_WIDTH<50
#undef MAX_WIDTH
#define MAX_WIDTH 50
#else
#undef MAX_WIDTH
#define MAX_WIDTH 100
#endif
char str[MAX_WIDTH];
註意看這一連串的指令 #if, #elsif 和 #else 是怎樣以 #endif 結尾的。
#line
當我們編譯一段程式的時候,如果有錯誤發生,編譯器會在錯誤前面顯示出錯文件的名稱以及文件中的第幾行發生的錯誤。
指令#line 可以使我們對這兩點進行控制,也就是說當出錯時顯示文件中的行數以及我們希望顯示的文件名。它的格式是:
#line number "filename"這裡number 是將會賦給下一行的新行數。它後面的行數從這一點逐個遞增。
filename 是一個可選參數,用來替換自此行以後出錯時顯示的文件名,直到有另外一個#line指令替換它或直到文件的末尾。例如:
#line 1 "assigning variable"int a?;
這段代碼將會產生一個錯誤,顯示為在文件"assigning variable", line 1 。
#error
這個指令將中斷編譯過程並返回一個參數中定義的出錯信息,例如:
#ifndef __cplusplus#error A C++ compiler is required
#endif
這個例子中如果 __cplusplus 沒有被定義就會中斷編譯過程。
#include
這個指令我們已經見到很多次。當預處理器找到一個#include 指令時,它用指定文件的全部內容替換這條語句。聲明包含一個文件有兩種方式:
#include "file"#include <file>
兩種表達的唯一區別是編譯器應該在什麼路經下尋找指定的文件。第一種情況下,文件名被寫在雙引號中,編譯器首先在包含這條指令的文件所在的目錄下進行尋找,如果找不到指定文件,編譯器再到被配置的預設路徑下(也就是標準頭文件路徑下)進行尋找。
如果文件名是在尖括弧 <> 中,編譯器會直接到預設標準頭文件路徑下尋找。
#pragma
這個指令是用來對編譯器進行配置的,針對你所使用的平臺和編譯器而有所不同。要瞭解更多信息,請參考你的編譯器手冊。
如果你的編譯器不支持某個#pragma的特定參數,這個參數會被忽略,不會產生出錯。
預定義的巨集名稱 (Predefined macro names)
以下巨集名稱在任何時候都是定義好的:
macro | value |
---|---|
__LINE__ | 整數值,表示當前正在編譯的行在源文件中的行數。 |
__FILE__ | 字元串,表示被編譯的源文件的文件名。 |
__DATE__ | 一個格式為 "Mmm dd yyyy" 的字元串,存儲編譯開始的日期。 |
__TIME__ | 一個格式為 "hh:mm:ss" 的字元串,存儲編譯開始的時間。 |
__cplusplus | 整數值,所有C++編譯器都定義了這個常量為某個值。如果這個編譯器是完全遵守C++標準的,它的值應該等於或大於199711L,具體值取決於它遵守的是哪個版本的標準。 |
例如:
// 標準巨集名稱
#include <iostream>
using namespace std;
int main()
{
cout << "This is the line number "
<< __LINE__;
cout << " of file " << __FILE__
<< ".\n";
cout << "Its compilation began "
<< __DATE__;
cout << " at " << __TIME__ << ".\n";
cout << "The compiler gives a "
<<"__cplusplus value of "
<< __cplusplus;
return 0;
}
輸出結果:
This is the line number 7 of file /home/jay/stdmacronames.cpp.Its compilation began Nov 1 2005 at 10:12:29.
The compiler gives a __cplusplus value of 1