忘了啥時候記的筆記了
@
宏定義
宏定義是C提供的三種預處理功能的其中一種,這三種預處理包括:宏定義、文件包含、條件編譯
1.不帶參數的宏定義:
宏定義又稱為宏代換、宏替換,簡稱“宏”。 格式: #define 標識符 字符串 ,其中的標識符就是所謂的符號常量,也稱為“宏名”。
預處理(預編譯)工作也叫做宏展開:將宏名替換為字符串。 掌握"宏"概念的關鍵是“換”。一切以換為前提、做任何事情之前先要換,准確理解之前就要“換”。 即在對相關命令或語句的含義和功能作具體分析之前就要換:
例: #define PI 3.1415926 ,把程序中出現的PI全部換成3.1415926
說明:
(1)宏名一般用大寫
(2)使用宏可提高程序的通用性和易讀性,減少不一致性,減少輸入錯誤和便於修改。例如:數組大小常用宏定義
(3)預處理是在編譯之前的處理,而編譯工作的任務之一就是語法檢查,預處理不做語法檢查。
(4)宏定義末尾不加分號;
(5)宏定義寫在函數的花括號外邊,作用域為其后的程序,通常在文件的最開頭。
(6)可以用#undef命令終止宏定義的作用域
(7)宏定義可以嵌套
(8)字符串" "中永遠不包含宏
(9)宏定義不分配內存,變量定義分配內存。
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)宏展開不占運行時間,只占編譯時間,函數調用占運行時間(分配內存、保留現場、值傳遞、返回值)
C語言宏定義技巧(常用宏定義)
寫好C語言,漂亮的宏定義很重要,使用宏定義可以防止出錯,提高可移植性,可讀性,方便性
1,防止一個頭文件被重復包含
#ifndef TEST_H // 1
#define TEST_H // 2
//.......... // 一些聲明語句
#endif // 3
以上代碼:如果沒有定義TEST_H變量,就繼續往下執行,如果定義了,就直接endif,即不做任何操作。
如果一個文件中包含了兩個同樣的頭文件,那么第一次調用這個頭文件的時候,沒有定義TEST_H,經過1,到達2,定義TEST_H,進行代碼段的執行,到3結束。但是當第二次調用頭文件時,已經定義了TEST_H,所以就不會進行下面的代碼聲明了。這就是條件編譯進行防止頭文件包含的原理
define后面只要寫個"文件名"保證其是一個獨一無二的定義就可以了
2,重新定義一些類型,防止由於各種平台和編譯器的不同,而產生的類型字節數差異,方便移植。
typedef unsigned char boolean; /* Boolean value type. */
typedef unsigned long int uint32; /* Unsigned 32 bit value */
typedef unsigned short uint16; /* Unsigned 16 bit value */
typedef unsigned char uint8; /* Unsigned 8 bit value */
typedef signed long int int32; /* Signed 32 bit value */
typedef signed short int16; /* Signed 16 bit value */
typedef signed char int8; /* Signed 8 bit value */
當時寫單片機的時候常用這東西。
3,得到指定地址上的一個字節或字
#define MEM_B( x ) ( *( (byte *) (x) ) )
#define MEM_W( x ) ( *( (word *) (x) ) )
這東西我沒咋用過。倒是當年寫匯編的時候會直接對內存地址進行操作,c我沒這么用過,主要是不知道確切地址,
4,求最大值和最小值
#define MAX( x, y ) ( ((x) > (y)) ? (x) : (y) )
#define MIN( x, y ) ( ((x) < (y)) ? (x) : (y) )
三目運算符代替函數,算是常用的一個操作。
5,得到一個變量的地址(word寬度)
#define B_PTR( var ) ( (byte *) (void *) &(var) )
#define W_PTR( var ) ( (word *) (void *) &(var) )
6,得到一個字的低位和高位字節
#define WORD_LO(xxx) ((byte) ((word)(xxx) & 255)) //得到低位
#define WORD_HI(xxx) ((byte) ((word)(xxx) >> 8)) //得到高位
7,將一個字母轉換為大寫
#define UPCASE( c ) ( ((c) >= 'a' && (c) <= 'z') ? ((c) - 0x20) : (c) )
當時比較菜
8,返回數組元素的個數
#define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
也是個常用的方法,大一的時候比較菜,投機取巧嘛
9,宏定義防止使用是錯誤
用小括號包含。例如
#define ADD(a,b) (a+b)
可以看到上面的一些宏定義看上去有些是不必要的小括號,其實都是一次次試出來的,都是必要的
