C 高階篇(五) 預處理指令

2021-07-13 19:29:01 字數 3309 閱讀 5238

預處理指令是我們寫在程式**中的給預處理器(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

#define max_width 200

char str2[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) 用來使得其後面所跟的程式部分只有在特定條件下才被編譯。這些條件只能夠是常量表示式,例如:

#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 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

兩種表達的唯一區別是編譯器應該在什麼路經下尋找指定的檔案。第一種情況下,檔名被寫在雙引號中,編譯器首先在包含這條指令的檔案所在的目錄下進行尋找,如果找不到指定檔案,編譯器再到被配置的預設路徑下(也就是標準標頭檔案路徑下)進行尋找。

如果檔名是在尖括號 <> 中,編譯器會直接到預設標準標頭檔案路徑下尋找。

如果你的編譯器不支援某個#pragma的特定引數,這個引數會被忽略,不會產生出錯。

以下巨集名稱在任何時候都是定義好的:

macro

value

__line__

整數值,表示當前正在編譯的行在原始檔中的行數。

__file__

字串,表示被編譯的原始檔的檔名。

__date__

乙個格式為 "mmm dd yyyy" 的字串,儲存編譯開始的日期。

__time__

乙個格式為 "hh:mm:ss" 的字串,儲存編譯開始的時間。

__cplusplus

整數值,所有c++編譯器都定義了這個常量為某個值。如果這個編譯器是完全遵守c++標準的,它的值應該等於或大於199711l,具體值取決於它遵守的是哪個版本的標準。

例如:

// 標準巨集名稱

#include

using

namespace std;

int main()

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

C 預處理指令

1.define 通常和 if一起使用 使用 define可以定義乙個符號,並通過將該符號用作表示式傳遞給 if 指令,使該表示式的計算結果為true 比如 preprocessor if.cs define debug define vc v7 using system public class ...

C預處理指令

一 預處理的由來 在c 的歷史發展中,有很多的語言特徵 特別是語言的晦澀之處 來自於c語言,預處理就是其中的乙個。c 從c語言那裡把c語言預處理器繼承過來 c語言預處理器,被bjarne博士簡稱為cpp,不知道是不是c program preprocessor的簡稱 二 常見的預處理功能 預處理器的...

C 預處理指令

巨集定義指令定義了乙個識別符號 巨集名 及乙個字串,在源程式中每次遇到該識別符號時,均以定義的字串替換它 巨集替換 巨集名和引數間不能有空格,識別符號和字串間可有任意空格 字串中應習慣對巨集引數加上括號,這樣使用巨集時,如果實參是表示式,則不易出錯 巨集替換只作巨集名和引數替換,不做計算,不做表示式...