宏定義數組
宏定義是C提供的三種預處理功能的其中一種,這三種預處理包括:宏定義、文件包含、條件編譯
1.
不帶參數的宏定義:
宏定義又稱爲宏代換、宏替換,簡稱「宏」。
格式:
#define標識符 字符串ide
其中的標識符就是所謂的符號常量,也稱爲「宏名」。
預處理(預編譯)工做也叫作宏展開:將宏名替換爲字符串。
掌握"宏"概念的關鍵是「換」。一切以換爲前提、作任何事情以前先要換,準確理解以前就要「換」。
即在對相關命令或語句的含義和功能做具體分析以前就要換:
例:
#define PI 3.1415926
把程序中出現的PI所有換成3.1415926
說明:
(1)宏名通常用大寫
(2)使用宏可提升程序的通用性和易讀性,減小不一致性,減小輸入錯誤和便於修改。例如:數組大小經常使用宏定義
(3)預處理是在編譯以前的處理,而編譯工做的任務之一就是語法檢查,預處理不作語法檢查。
(4)宏定義末尾不加分號;
(5)宏定義寫在函數的花括號外邊,做用域爲其後的程序,一般在文件的最開頭。
(6)能夠用#undef命令終止宏定義的做用域
(7)宏定義容許嵌套
(8)字符串" "中永遠不包含宏函數
(9)宏定義不分配內存,變量定義分配內存。
(10)宏定義不存在類型問題,它的參數也是無類型的。
2.
帶參數的宏定義:
除了通常的字符串替換,還要作參數代換
格式:
#define 宏名(參數表) 字符串
例如:#define S(a,b) a*b
area=S(3,2);第一步被換爲area=a*b; ,第二步被換爲area=3*2;
相似於函數調用,有一個啞實結合的過程:
(1)實參若是是表達式容易出問題
#define S(r) r*r
area=S(a+b);第一步換爲area=r*r;,第二步被換爲area=a+b*a+b;
正確的宏定義是#define S(r) ((r)*(r))
(2)宏名和參數的括號間不能有空格
(3)宏替換隻做替換,不作計算,不作表達式求解
(4)函數調用在編譯後程序運行時進行,而且分配內存。宏替換在編譯前進行,不分配內存
(5)宏的啞實結合不存在類型,也沒有類型轉換。
(6)函數只有一個返回值,利用宏則能夠設法獲得多個值
(7)宏展開使源程序變長,函數調用不會
(8)宏展開不佔運行時間,只佔編譯時間,函數調用佔運行時間(分配內存、保留現場、值傳遞、返回值)
3.
宏定義其餘冷門、重點知識
#define用法
一、 用無參宏定義一個簡單的常量
#define LEN 12
這個是最多見的用法,但也會出錯。
好比下面幾個知識點你會嗎?能夠看下:
(1) #define NAME "zhangyuncong"
程序中有"NAME"則,它會不會被替換呢?
(2) #define 0x abcd
能夠嗎?也就是說,可不能夠用把標識符的字母替換成別的東西?
(3) #define NAME "zhang
這個能夠嗎?
(4) #define NAME "zhangyuncong"
程序中有上面的宏定義,而且,程序裏有句:
NAMELIST這樣,會不會被替換成"zhangyuncong"LIST
四個題答案都是十分明確的。
第一個,""內的東西不會被宏替換。這一點應該大都知道。
第二個,宏定義前面的那個必須是合法的用戶標識符
第三個,宏定義也不是說後面東西隨便寫,不能把字符串的兩個""拆開。
第四個:只替換標識符,不替換別的東西。NAMELIST總體是個標識符,而沒有NAME標識符,因此不替換。
也就是說,這種狀況下記住:#define 第一位置第二位置
(1) 不替換程序中字符串裏的東西。
(2) 第一位置只能是合法的標識符(能夠是關鍵字)
(3) 第二位置若是有字符串,必須把""配對。
(4) 只替換與第一位置徹底相同的標識符
還有就是老生常談的話:記住這是簡單的替換而已,不要在中間計算結果,必定要替換出表達式以後再算。
二、 帶參宏通常用法
好比#define MAX(a,b) ((a)>(b)?(a):(b))
則遇到MAX(1+2,value)則會把它替換成:
((1+2)>(value)?(1+2):(value))
注意事項和無參宏差很少。
但仍是應注意
#define FUN(a) "a"htm
則,輸入FUN(345)會被替換成什麼?
其實,若是這麼寫,不管宏的實參是什麼,都不會影響其被替換成"a"的命運。
也就是說,""內的字符不被當成形參,即便它和如出一轍。
那麼,你會問了,我要是想讓這裏輸入FUN(345)它就替換成"345"該怎麼實現呢?
請看下面關於#的用法
三、 有參宏定義中#的用法
#define STR(str) #str內存
#用於把宏定義中的參數兩端加上字符串的""
好比,這裏STR(my#name)會被替換成"my#name"
通常由任意字符均可以作形參,但如下狀況會出錯:
STR())這樣,編譯器不會把「)」當成STR()的參數。
STR(,)同上,編譯器不會把「,」當成STR的參數。
STR(A,B)若是實參過多,則編譯器會把多餘的參數捨去。(VC++2008爲例)
STR((A,B))會被解讀爲實參爲:(A,B),而不是被解讀爲兩個實參,第一個是(A第二個是B)。
四、 有參宏定義中##的用法
#define WIDE(str) L##str作用域
則會將形參str的前面加上L
好比:WIDE("abc")就會被替換成L"abc"
若是有#define FUN(a,b) vo##a##b()
那麼FUN(id ma,in)會被替換成void main()
五、 多行宏定義:
#define doit(m,n) for(int i=0;i<(n);++i)\字符串
{\
m+=i;\
}