C語言超全面define預(yù)處理指令的使用說明
前言
C語言中源代碼到可執(zhí)行文件的第一階段,也就是預(yù)處理階段,會(huì)檢查源文件中的預(yù)處理指令語句和宏定義,并對(duì)源代碼進(jìn)行相應(yīng)的替換,預(yù)處理過程還會(huì)刪除程序中的注釋和多余的空白符號(hào)。
預(yù)處理指令是以#開頭的代碼行,#必須是該行除了空白符外的第一個(gè)字符,#后是指令關(guān)鍵字,在#和指令關(guān)鍵字之間允許存在若干個(gè)空白字符,define是宏定義命令。在C語言程序中允許用一個(gè)標(biāo)識(shí)符來表示一個(gè)字符串,稱為“宏”,“宏”又分為有參和無參,有參又稱為“宏函數(shù)”,被定義為“宏”的標(biāo)識(shí)符稱為“宏名”。
#define 定義宏(無參)
語法規(guī)定:
#define name stuff
name:標(biāo)識(shí)符名\宏名
stuff:可以是關(guān)鍵字、常量、關(guān)鍵字、標(biāo)識(shí)符、標(biāo)點(diǎn)符號(hào)、運(yùn)算符,表達(dá)式
在預(yù)處理階段,編譯器會(huì)在程序中使用#define定義的標(biāo)識(shí)符替換成stuff,可以通過預(yù)處理生成的.i文件查看效果。
//stuff是數(shù)值常量 #define NUM 10

//stuff是關(guān)鍵字 #define reg register

//stuff是標(biāo)點(diǎn)符號(hào) #define GREATER_THAN >

//stuff的更多表達(dá)方式
#define do_forever for(;;)
//若定義的stuff過長,可以分成幾行寫,除了最后一行外,每行的后面都加一個(gè)\(續(xù)行符)
#define DBBUG_PRINT printf("file:%s\tline:%d\t \
date:%s\ttime:%s\n",\
__FILE__,__LINE__,\
__DATE__,__TIME__)總結(jié):#define定義的name(宏名),在預(yù)編譯階段會(huì)將所有的宏名替換成stuff,stuff內(nèi)容被替換到源代碼中。稱為“宏代換”或“宏展開”。
注意:define定義的標(biāo)識(shí)符的時(shí)候,后面加上“;”會(huì)將“;”認(rèn)為是stuff中的內(nèi)容。
#define 定義宏函數(shù)
宏函數(shù)的申明方式:#define name(parament-list) stuff
parament-list:參數(shù)列表
注意:參數(shù)列表的左括號(hào)必須與name緊鄰,如果兩者之間有空白存在,參數(shù)列表就會(huì)被解釋為stuff的部分。
宏函數(shù)存在的問題1
#include <stdio.h>
#define SQUARE(x) x*x//定義一個(gè)宏函數(shù)求平方
int main()
{
int x = SQUARE(3+1);//替換后x的計(jì)算結(jié)果是多少?答案是:7
}為什么呢?

在給宏函數(shù)傳參時(shí),如果傳遞的是一個(gè)表達(dá)式,不會(huì)先計(jì)算表達(dá)式的結(jié)果再進(jìn)行傳參,而是直接將表達(dá)式整體作為參數(shù)傳遞。
那么如何防止發(fā)生這樣的情況呢?+()

宏函數(shù)存在的問題2
#include <stdio.h>
#define SUM(x,y) (x)+(y)
int main()
{
int a = 10;
int b = 5;
int c = SUM(a,b)*2;//替換后c的結(jié)果為20,why
return 0;
}我們看看替換后的結(jié)果

這又該如何解決呢?

總結(jié):在對(duì)數(shù)值表達(dá)式進(jìn)行求值的宏定義應(yīng)該用這兩種方式加上括號(hào),避免在使用宏參數(shù)的操作符或鄰近操作符之間不可預(yù)料的相互作用。
#define替換規(guī)則:
1.在使用宏函數(shù)時(shí),首先對(duì)參數(shù)進(jìn)行檢查,看看參數(shù)中是否包含任何#define定義的標(biāo)識(shí)符,如果有,他們首先被替換。
2.替換的內(nèi)容被插入到源文件原來的位置。對(duì)于宏函數(shù),參數(shù)名被他們的值替換
宏的更多規(guī)則特性
1.宏名一般用大寫
2.使用宏可提高程序的通用性和易讀性,便于修改。
3.宏定義末尾不加分號(hào)
4.宏定義寫在函數(shù)的大括號(hào)外面,作用域?yàn)槠浜蟮某绦?,通常放在開頭
5.宏函數(shù)不可遞歸
6.宏定義不分配內(nèi)存,變量定義分配內(nèi)存
7.字符串" "中永遠(yuǎn)不包含宏
8.宏定義不存在類型問題,他的參數(shù)也沒有類型
宏的缺點(diǎn)
1.宏不能調(diào)試
2.宏由于與類型無關(guān),不夠嚴(yán)謹(jǐn)
3.宏可能帶來運(yùn)算符優(yōu)先級(jí)的問題,導(dǎo)致容易出錯(cuò)
常見預(yù)處理指令
#define:宏定義
#undef:撤銷已經(jīng)定義過的宏名
#include:將另一個(gè)源文件嵌入到#include源文件中
#if~#endif:如果#if后面的常量表達(dá)式為真,則編譯#if~#endif之間的代碼,如果為假,跳過這些代碼不編譯。
#if~#elif~#else~#endif:和if~else if~else類似,可以建立更分支。
#ifdef symbol~endif:判斷是否被定義,定義了編譯他們之間內(nèi)容
#ifndef symbol~endif:判斷是否被定義,沒定義編譯他們之間的內(nèi)容
#line:改變當(dāng)前行數(shù)和文件名稱,是在編譯程序中預(yù)先定義的標(biāo)識(shí)符命令的基本形式:#line number["filename"]
#error:編譯程序時(shí),只要遇到#error就會(huì)生成一個(gè)編譯錯(cuò)誤的提示信息,并停止編譯。
#pragma: 可以設(shè)定編譯程序完成一些特點(diǎn)的動(dòng)作(可以通過編譯程序的菜單中設(shè)置),可以向編譯程序傳送各種指令。
到此這篇關(guān)于C語言超全面define預(yù)處理指令的使用說明的文章就介紹到這了,更多相關(guān)C語言define內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C語言一級(jí)指針二級(jí)指針和三級(jí)指針區(qū)別及使用詳解
這篇文章主要為大家介紹了C語言一級(jí)指針二級(jí)指針和三級(jí)指針,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05
Java C++ 算法leetcode828統(tǒng)計(jì)子串中唯一字符乘法原理
這篇文章主要為大家介紹了Java C++ 算法leetcode828統(tǒng)計(jì)子串中唯一字符乘法原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-09-09
C++重載運(yùn)算符實(shí)現(xiàn)分?jǐn)?shù)加減乘除
這篇文章主要為大家詳細(xì)介紹了C++重載運(yùn)算符實(shí)現(xiàn)分?jǐn)?shù)加減乘除,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-06-06

