轉 http://blog.csdn.net/pachonghanya/article/details/6972632
字符串常量定義時的換行問題
構建較長的字符串是續行的常見用途, 還有一個作用是定義跨行的宏。
如果我們不使用反斜杠,當我們試圖初始化一個跨多行的字符串是,c語言編譯器就會發出警告。如下面的語句所示:
char letters[] = {"abcdefghijklmnopqrstuvwxyz
ABCDEFGHIJKLMNOPQRSTUVWXYZ"};
但是我們在行尾使用反斜杠, 那么就可以吧字符串常量跨行書寫, 如下所示:
char letters[] = {"abcdefghijklmnopqrstuvwxyz\
ABCDEFGHIJKLMNOPQRSTUVWXYZ"};
並將其初始化為如下的初值:"abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ"
c語言中還有一種拆分字符串的方法,那就是將其寫個多個相鄰的字符串。這些字符串之間用0個或者多個空白、制作符以及換行符隔開。c語言編譯器會自動將這些字符串連接起來。因此,下面的表達式:"one" "two" "three" 實際上相當於 "onetwothree".
char letters[] = {"abcdefghijklmnopqrstuvwxyz"
"ABCDEFGHIJKLMNOPQRSTUVWXYZ"}
無參數的宏定義的一般形式為
# define 標識符 字符序列
其中# define之后的標識符稱為宏定義名(簡稱宏名),要求宏名與字符序列之間用空格符分隔。這種宏定義要求編譯預處理程序將源程序中隨后所有的定名的出現(注釋與字符串常量中的除外)均用字符序列替換之。前面經常使用的定義符號常量是宏定義的最簡單應用。如有:
# define TRUE 1
# define FALSE 0
則在定義它們的源程序文件中,凡定義之后出現的單詞TRUE將用1替代之;出現單詞FALSE將用0替代之。
在宏定義的#之前可以有若干個空格、制表符,但不允許有其它字符。宏定義在源程序中單獨另起一行, 換行符是宏定義的結束標志。如果一個宏定義太長,一行不夠時, 可采用續行的方法。續行是在鍵人回車符之前先鍵入符號""。注意回車要緊接在符號""之后,中間不能插入其它符號。
無參數的宏定義的一般形式為
# define 標識符 字符序列
其中# define之后的標識符稱為宏定義名(簡稱宏名),要求宏名與字符序列之間用空格符分隔。這種宏定義要求編譯預處理程序將源程序中隨后所有的定名的出現(注釋與字符串常量中的除外)均用字符序列替換之。前面經常使用的定義符號常量是宏定義的最簡單應用。如有:
# define TRUE 1
# define FALSE 0
則在定義它們的源程序文件中,凡定義之后出現的單詞TRUE將用1替代之;出現單詞FALSE將用0替代之。
在宏定義的#之前可以有若干個空格、制表符,但不允許有其它字符。宏定義在源程序中單獨另起一行,換行符是宏定義的結束標志。如果一個宏定義太長,一行不夠時, 可采用續行的方法。續行是在鍵人回車符之前先鍵入符號""。注意回車要緊接在符號""之后,中間不能插入其它符號。
宏定義的有效范圍稱為宏定義名的轄域, 轄域從宏定義的定義結束處開始到其所在的源程序文件末尾。宏定義名的轄域不受分程序結構的影響。可以用預處理命令#undef終止宏定義名的轄域。
在新的宏定義中,可以使用前面已定義的宏名。例如,
# define R 2.5
# define PI 3.1415926
# define Circle 2*PI*R
# define Area PI* R * R
程序中的Circle被展開為2*3.1415926* 2.5, Area被展開為3.1415926*2.5*2.5。
如有必要,宏名可被重復定義。被重復定義后,宏名原先的意義被新意義所代替。
通常,無參數的宏定義多用於定義常量。程序中統一用宏名表示常量值,便於程序前后統一,不易出錯,也便於修改,能提高程序的可讀性和可移植性。特別是給數組元素個數一個宏定義,並用宏名定義數組元素個數能部分彌補數組元素個數固定的不足。
注意:預處理程序在處理宏定義時,只作字符序列的替換工作,不作任何語法的檢查。如果宏定義不當,錯誤要到預處理之后的編譯階段才能發現。宏定義以換行結束,不需要分號等符號作分隔符。如有以下定定義:
# define PI 3.1415926;
原希望用PI求圓的周長的語句
c=2*PI*r;
經宏展開后,變成
c=2*3.1415926*r;
這就不能達到希望的要求。
帶參數宏定義進一步擴充了無參數宏定義的能力,在字符序列替換同時還能進行參數替換。帶參數定定義的一般形式為
# define 標識符(參數表)字符序列
其中參數表中的參數之間用逗號分隔,字符序列中應包含參數表中的參數。在定義帶參數的宏時,宏名標識符與左圓括號之間不允許有空白符,應緊接在一起,否則變成了無參數的宏定義。如有宏定義:
# define MAX(A,B) ((A) > (B)?(A):(B))
則代碼 y= MAX( p+q, u+v)將被替換成 y=((p+q) >(u+v)?(p+q):(u+v)。
程序中的宏調用是這樣被替換展開的,分別用宏調用中的實在參數字符序列(如p+q和u+V) 替換宏定義字符序列中對應所有出現的形式參數(如用p+q替代所有形式參數A,用u+V替代所有形式參數B),而宏定義字符序列中的不是形式參數的其它字符則保留。這樣形成的字符序列,即為宏調用的展開替換結果。宏調用提供的實在參數個數必須與宏定義中的形式參數個數相同。
注意:宏調用與函數調用的區別。函數調用在程序運行時實行,而宏展開是在編譯的預處理階段進行;函數調用占用程序運行時間,宏調用只占編譯時間;函數調用對實參有類型要求,而宏調用實在參數與宏定義形式參數之間沒有類型的概念,只有字符序列的對應關系。函數調用可返回一個值,宏調用獲得希望的C代碼。另外,函數調用時,實參表達式分別獨立求值在前,執行函數體在后。宏調用是實在參數字符序列替換形式參數。替換后,實在參數字符序列就與相鄰的字符自然連接,實在參數的獨立性就不一定依舊存在。如下面的宏定義:
# define SQR(x) x*x
希望實現表達式的平方計算。對於宏調用
P=SQR(y)
能得到希望的宏展開p= y*y。但對於宏調用q=SQR(u+v)得到的宏展開是q=u+V*u+V。顯然,后者的展開結果不是程序設計者所希望的。為能保持實在參數替換后的獨立性,應在宏定義中給形式參數加上括號。進一步,為了保證宏調用的獨立性,作為算式的宏定義也應加括
號。如 SQR宏定義改寫成:
# define SQR((x)*(x))
才是正確的宏定義。
對於簡短的表達式計算函數,或為了提高程序的執行效率、避免函數調用時的分配存儲單元、保留現場、參數值傳遞、釋放存儲單元等工作。可將函數定義改寫成宏定義。所以合理使用宏定義,可以使程序更簡潔。
