帶大家逐個(gè)了解學(xué)習(xí)C語(yǔ)言預(yù)處理命令
C語(yǔ)言的源代碼中可以包括各種編譯指令,這些指令就稱之為預(yù)處理命令。雖然它們實(shí)際上不是c語(yǔ)言的一部分,但卻擴(kuò)展了c程序設(shè)計(jì)的環(huán)境,預(yù)處理程序和注釋簡(jiǎn)化程序開(kāi)發(fā)過(guò)程,并且提高程序的可讀性。然而這些命令是在編譯前由預(yù)處理程序?qū)υ次募念A(yù)處理文件進(jìn)行加工。
預(yù)處理是在c語(yǔ)言編譯的4個(gè)階段(預(yù)處理、編譯、匯編、鏈接)的第一個(gè)階段。
C語(yǔ)言的預(yù)處理功能有一下3種:
1,宏定義
2,文件包含
3,條件編譯
分別用宏定義命令、文件包含命令、條件編譯命令實(shí)現(xiàn),為了與其他的c語(yǔ)句區(qū)別,預(yù)處理命令以符號(hào)#開(kāi)頭。
下面領(lǐng)著大家逐個(gè)了解學(xué)習(xí)這些預(yù)處理命令。
一:宏定義:
不帶參數(shù)的宏定義
一般形式:
#define 標(biāo)識(shí)符 字符串
作用:用一個(gè)特定的標(biāo)識(shí)符來(lái)代表一個(gè)字符串。
如:#define PI 3.1415926
在編譯處理時(shí),把程序中在該命令之后的所有PI都用“3.1415926”替換。這個(gè)標(biāo)識(shí)符(名字)稱為“宏名”,在預(yù)編譯時(shí)將宏名替換成字符串的過(guò)程叫做“宏展開(kāi)”。
注意:
1,宏名一般習(xí)慣用大寫(xiě)來(lái)表示;
2,用宏名代替一個(gè)字符串,可以減少在程序書(shū)寫(xiě)的過(guò)多的繁瑣的事件,當(dāng)需要某一個(gè)常量時(shí),只需要改變#define命令即可;
3,宏定義簡(jiǎn)單的說(shuō)就是置換,不做正確性檢查,假如寫(xiě)錯(cuò)了,預(yù)處理也會(huì)照樣帶入,只有在編譯已被宏展開(kāi)的源程序時(shí)才報(bào)錯(cuò);
4,宏定義不是語(yǔ)句,不需要分號(hào);
5,宏定義程序中函數(shù)的外部,作用范圍是在定義之后到本源程序結(jié)束;
6,可以使用#undef命令終止宏定義作用域;例如:
#define H 9.9
main()
{
...........; }
#undef H
gg()
{
.............;}
以上可以看出:H的作用域就只有在main起作用。
7,宏定義時(shí),可以使用已經(jīng)定義的宏定義,可以層層置換;
8,對(duì)程序中用雙引號(hào)括起來(lái)的字符,即使與宏名相同,也不會(huì)進(jìn)行置換(printf()函數(shù)的格式化輸出);
9,宏定義與定義變量含義不一樣,只做字符替換,不會(huì)分配空間;
帶參數(shù)的宏:
定義的一般形式:
#define 宏名(參數(shù)列表) 字符串
例如:
注意:
1,對(duì)帶參數(shù)的宏展開(kāi),只是將語(yǔ)句中的宏名后面的括號(hào)內(nèi)的實(shí)參字符串代替#define命令行中的參數(shù);
2,宏定義時(shí),宏名與帶參數(shù)的括號(hào)不得有空格;
帶參數(shù)的宏定義與函數(shù)的區(qū)別:
函數(shù)調(diào)用時(shí),先求出實(shí)參表達(dá)式的值,然后帶形參;
而使用帶參數(shù)的宏只是進(jìn)行簡(jiǎn)單的字符替換;
函數(shù)調(diào)用是在程序運(yùn)行時(shí)處理的,分配臨時(shí)的內(nèi)存單元;
而宏展開(kāi)是在編譯時(shí)進(jìn)行的,在展開(kāi)時(shí)并分配內(nèi)存單元,不進(jìn)行值的傳遞處理,也沒(méi)有“返回值”的概念;
函數(shù)中的實(shí)參和形參都要定義類型,二者類型要求一致,若不一致,應(yīng)進(jìn)行類型轉(zhuǎn)換;
而宏不存在類型問(wèn)題,宏名無(wú)類型,參數(shù)也無(wú)類型,展開(kāi)時(shí)帶入指定的字符即可。宏定義時(shí),字符串可以是任何類型的數(shù)據(jù);
調(diào)用函數(shù)可以得到一個(gè)返回值;
而用宏可以設(shè)法得到幾個(gè)結(jié)果;
使用宏次數(shù)多時(shí),展開(kāi)后源程序長(zhǎng);
而函數(shù)調(diào)用不會(huì)使源程序變長(zhǎng);
宏替換不占運(yùn)行時(shí)間,只占編譯時(shí)間;
而函數(shù)調(diào)用則占運(yùn)行時(shí)間(分配單元、值傳遞、返回等等);
二:文件包含
含義:指將其他文件的內(nèi)容全部包括進(jìn)來(lái),即將另外的文件內(nèi)容包含到本文件中。
用法:
#include
或#include”文件名”
#include
使用<>和""的區(qū)別:使用尖括號(hào)時(shí),系統(tǒng)到存放C庫(kù)函數(shù)的頭文件的目錄中去找要包含的文件,這稱為標(biāo)準(zhǔn)方式。使用引號(hào)時(shí),系統(tǒng)優(yōu)先在用戶當(dāng)前目錄下查找要包含的文件,若找不到,再按標(biāo)準(zhǔn)方式查找。
說(shuō)明:
1,一個(gè)include命令只能指定一個(gè)被包含文件,若要包含n個(gè)文件,要用n個(gè)include命令;
2,如果一個(gè)文件包含了1.h,而文件1.h用到了3.h的內(nèi)容,則可在文件中如下定義:
#include“1.h”
#include“2.h”
3,文件包含可以嵌套,即在一個(gè)被包含文件,又可以包含另一個(gè)文件
三:條件編譯
一般情況下,源程序中所有的文件都要參加編譯。但有時(shí)只需要程序的一部分代碼進(jìn)行編譯,這時(shí)就要指定編譯條件,這就是條件編譯。
條件編譯有以下幾種形式:
用法1:
#ifdef 標(biāo)識(shí)符
程序段1
#else
程序段2
#endif
說(shuō)明:當(dāng)定義過(guò)標(biāo)識(shí)符,則編譯程序段1,否則編譯程序段2。#else部分可以省略,即:
#ifdef
程序段1
#endif
用法2:
#ifndef 標(biāo)識(shí)符
程序段1
#else
程序段2
#endif
說(shuō)明:與上一種類似,只不過(guò)將ifdef換成ifndef。當(dāng)未定義過(guò)標(biāo)識(shí)符,則編譯程序段1,否則編譯程序段2。#else部分可以省略
用法3:
#if 表達(dá)式
程序段1
#else
程序段2
#endif
說(shuō)明:當(dāng)指定的表達(dá)式是真(非零)的時(shí)候,編譯程序段1,否則編譯程序段2。#else部分可以省略