預處理指令提供按條件跳過原始檔中的節、報告錯誤和警告條件,以及描繪源**的不同區域的能力。使用術語「預處理指令」只是為了與 c 和 c++ 程式語言保持一致。
預處理器的主要作用就是把通過預處理的內建功能對乙個資源進行等價替換,最常見的預處理有:檔案包含,條件編譯、布局控制和巨集替換4種。
[b]檔案包含[/b]:
#include 是一種最為常見的預處理,主要是做為檔案的引用組合源程式正文。
[b]條件編譯[/b]:
#if,#ifndef,#ifdef,#endif,#undef等也是比較常見的預處理,主要是進行編譯時進行有選擇的挑選,注釋掉一些指定的**,以達到版本控制、防止對檔案重複包含的功能。
[b]布局控制[/b]:
#progma,這也是我們應用預處理的乙個重要方面,主要功能是為編譯程式提供非常規的控制流資訊。
[b]巨集替換[/b]:
#define,這是最常見的用法,它可以定義符號常量、函式功能、重新命名、字串的拼接等各種功能。
[b]下面我們看一下常見的預處理指令[/b]:
#define 巨集定義
#undef 未定義巨集
#include 文字包含
#ifdef 如果巨集被定義就進行編譯
#ifndef 如果巨集未被定義就進行編譯
#endif 結束編譯塊的控制
#if 表示式非零就對**進行編譯
#else 作為其他預處理的剩餘選項進行編譯
#elif 這是一種#else和#if的組合選項
#line 改變當前的行數和檔名稱
#error 輸出乙個錯誤資訊
#pragma 為編譯程式提供非常規的控制流資訊
下面我們對這些預處理進行一一的說明,考慮到巨集的重要性和繁瑣性,我們把它放到最後講。
[b]檔案包含指令[/b]:
這種預處理使用方式是最為常見的,平時我們編寫程式都會用到,最常見的用法是:
#include //標準庫標頭檔案
#include //舊式的標準庫標頭檔案
#include "io.h" //使用者自定義的標頭檔案
#include "../file.h" //unix下的父目錄下的標頭檔案
#include "/usr/local/file.h" //unix下的完整路徑
#include "..\file.h" //dos下的父目錄下的標頭檔案
#include "\usr\local\file.h" //dos下的完整路徑
這裡面有2個地方要注意:
1、我們用還是?
我們主張使用,而不是,為什麼呢?我想你可能還記得我曾經給出過幾點理由,這裡我大致的說一下:首先,.h格式的標頭檔案早在98年9月份就被標準委員會拋棄了,我們應該緊跟標準,以適合時代的發展。其次,iostream.h只支援窄字符集,iostream則支援窄/寬字符集。
還有,標準對iostream作了很多的改動,介面和實現都有了變化。最後,iostream元件全部放入namespace std中,防止了名字汙染。
2、和"io.h"的區別?
其實他們唯一的區別就是搜尋路徑不同:
對於#include ,編譯器從標準庫路徑開始搜尋
對於#include "io.h" ,編譯器從使用者的工作路徑開始搜尋
[b]編譯控制指令[/b]:
這些指令的主要目的是進行編譯時進行有選擇的挑選,注釋掉一些指定的**,以達到版本控制、防止對檔案重複包含的功能。
使用格式,如下:
1、1. #ifdef identifier
2. your code
3. #endif
如果identifier為乙個定義了的符號,your code就會被編譯,否則剔除
2、1. #ifndef identifier
2. your code
3. #endif
如果identifier為乙個未定義的符號,your code就會被編譯,否則剔除
3、1. #if expression
2. your code
3. #endif
如果expression非零,your code就會被編譯,否則剔除
4、1. #ifdef identifier
2. your code1
3. #else
4. your code2
5. #endif
如果identifier為乙個定義了的符號,your code1就會被編譯,否則yourcode2就會被編譯
5、1. #if expressin1
2. your code1
3. #elif expression2 //呵呵,elif
4. your code2
5. #else
6. your code3
7. #enif
如果epression1非零,就編譯your code1,否則,如果expression2非零,就編譯your code2,否則,就編譯your code3
[b]其他預編譯指令[/b]
除了上面我們說的集中常用的編譯指令,還有3種不太常見的編譯指令:#line、#error、#pragma,我們接下來就簡單的談一下。
#line的語法如下:
1. #line number filename
例如:#line 30 a.h 其中,檔名a.h可以省略不寫。
這條指令可以改變當前的行號和檔名,例如上面的這條預處理指令就可以改變當前的行號為30,檔名是a.h。初看起來似乎沒有什麼用,不過,他還是有點用的,那就是用在編譯器的編寫中,我們知道編譯器對c++原始碼編譯過程中會產生一些中間檔案,通過這條指令,可以保證檔名是固定的,不會被這些中間檔案代替,有利於進行分析。
#error語法如下:
1. #error info
例如:1. #ifndef unix
2. #error this software requires the unix os.
3. #endif
這條指令主要是給出錯誤資訊,上面的這個例子就是,如果沒有在unix環境下,就會輸出this software requires the unix os.然後誘發編譯器終止。所以總的來說,這條指令的目的就是在程式崩潰之前能夠給出一定的資訊。
#pragma是非統一的,他要依靠各個編譯器生產者,例如,在sun c++編譯器中:
1. // 把name和val的起始位址調整為8個位元組的倍數
2. #progma align 8 (name, val)
3. char name[9];
4. double val;
5. //在程式執行開始,呼叫函式myfunction
6. #progma init (myfunction)
預定義識別符號
為了處理一些有用的資訊,預處理定義了一些預處理識別符號,雖然各種編譯器的預處理識別符號不盡相同,但是他們都會處理下面的4種:
1. __file__ 正在編譯的檔案的名字
2. __line__ 正在編譯的檔案的行號
3. __date__ 編譯時刻的日期字串,例如: "25 dec 2000"
4. __time__ 編譯時刻的時間字串,例如: "12:30:55"
例如:cout<<"the file is :"<<__file__"<<"! the lines is:"<<__line__<預處理何去何從
如何取代#include預處理指令,我們在這裡就不再一一討論了。
c++並沒有為#include提供替代形式,但是namespace提供了一種作用域機制,它能以某種方式支援組合,利用它可以改善#include的行為方式,但是我們還是無法取代#include。
#progma應該算是乙個可有可無的預處理指令,按照c++之父bjarne的話說,就是:"#progma被過分的經常的用於將語言語義的變形隱藏到編譯系統裡,或者被用於提供帶有特殊語義和笨拙語法的語言擴充。」
對於#ifdef,我們仍然束手無策,就算是我們利用if語句和常量表示式,仍然不足以替代她,因為乙個if語句的正文必須在語法上正確,滿足類檢查,即使他處在乙個絕不會被執行的分支裡面。
宣告:本文內容來自網路!
預處理基礎知識
預處理符號 file 當前檔案的原檔案名字 line 檔案當前的行號 data 檔案編譯的日期 time 檔案編譯的時間 sdtc 如果編譯器準尋ansic,其值為1,否則未定義 define巨集和 undef巨集 define name name 定義乙個巨集定義 undef 移除乙個巨集定義 和...
C C 預處理指令
預處理指令是以 號開頭的 行。號必須是該行除了任何空白字元外的第乙個字元。後是指令關鍵字,在關鍵字和 號之間允許存在任意個數的空白字元。整行語句構成了一條預處理指令,該指令將在編譯器進行編譯之前對源 做某些轉換。1.和 1 define pi 3.1415926 2 define paste n a...
C C 預處理指令
預處理指令 preprocessor directives define undef ifdef ifndef if endif else and elif line error include 預定義識別符號 pragma 預處理指令是我們寫在程式 中的給預處理器 preprocessor 的命令...