預處理器: 數組
編譯程序以前,先由預處理器檢查程序,根據程序中使用的預處理器指令,預處理器用符號縮略語所表明的內容替換程序中的縮略語。安全
預處理器能夠根據包含其餘文件,還能夠選擇讓編譯器處理哪些代碼,預處理器不能理解C,它通常是接受一些文本並將其轉換成其餘文本函數
1.#define 的用法spa
定義: .net
宏 #define 預處理指令 可出如今源文件的任何地方 指令定義的做用於從定義出現的位置開始直到文件的結尾。指針
預處理器在程序中發現了宏的實例後,總會用實體代替該宏,從宏變成最終的替換文本的過程稱爲宏展開。開發
預處理器發現程序中的宏後,會用它的等價替換文本代替宏。若是該字符還包括宏,則繼續替換這些宏。字符串
const關鍵字獲得了C的支持,這確實提供了一種建立常量的更靈活的方法,使用const能夠建立全局變量和局部變量、數字常量、數組常量和結構常量,另外一方面宏常量能夠用來指定標準數組大小並做爲const值得初始化值。get
#define LIMIT 20編譯器
const int LIM =50;
static int data1[LIMIT]; //合法
static int data2[LIM]; //無效
const int LIM2 = 2 * LIMIT; //合法
const int LIM3 = 2* LIM; // 無效
(1)簡單的用法
#define MAXTIME 1000
一個簡單的MAXTIME就定義好了,它表明1000,若是在程序裏面寫
if(i<MAXTIME){.........}
編譯器在處理這個代碼以前會對MAXTIME進行處理替換爲1000。
這樣的定義看起來相似於普通的常量定義CONST,但也有着不一樣,由於define的定義更像是簡單的文本替換,而不是做爲一個量來使用。
(2)函數定義(時刻注意#define只是進行文本替換 而不進行任何編譯處理)
define能夠像函數那樣接受一些參數,以下
#define max(x,y) (x)>(y)?(x):(y);
這個定義就將返回兩個數中較大的那個,看到了嗎?由於這個「函數」沒有類型檢查,就好像一個函數模板似的,固然,它絕對沒有模板那麼安全就是了。能夠做爲一個簡單的模板來使用而已。
可是這樣作的話存在隱患,例子以下:
#define Add(a,b) a+b;
在通常使用的時候是沒有問題的,可是若是遇到如:c * Add(a,b) * d的時候就會出現問題,代數式的本意是a+b而後去和c,d相乘,可是由於使用了define(它只是一個簡單的替換),因此式子實際上變成了
c*a + b*d
另外舉一個例子:
#define pin (int*);
pin a,b;
本意是a和b都是int型指針,可是實際上變成int* a,b;
a是int型指針,而b是int型變量。
這是應該使用typedef來代替define,這樣a和b就都是int型指針了。
因此咱們在定義的時候,養成一個良好的習慣,建議全部的層次都要加括號。
(3)#define 多行定義
define能夠替代多行的代碼,例如MFC中的宏定義(很是的經典,雖然讓人看了噁心)
#define MACRO(arg1, arg2) do { /
/* declarations */ /
stmt1; /
stmt2; /
/* ... */ /
} while(0) /* (no trailing ; ) */
關鍵是要在每個換行的時候加上一個"/"
(4).在大規模的開發過程當中,特別是跨平臺和系統的軟件裏,define 最重要的功能是條件編譯。
就是:
#ifdef WINDOWS
......
......
#endif
#ifdef LINUX
......
......
#endif
#ifdef XXX…(#else) …#endif
例如 #ifdef DV22_AUX_INPUT
#define AUX_MODE 3
#else
#define AUY_MODE 3
#endif
#ifndef XXX … (#else) … #endif
能夠在編譯的時候經過#define設置編譯環境
(5).如何定義宏、取消宏
//定義宏
#define [MacroName] [MacroValue]
//取消宏
#undef [MacroName]
普通宏
#define PI (3.1415926)
帶參數的宏
#define max(a,b) ((a)>(b)? (a),(b))
關鍵是十分容易產生錯誤,包括機器和人理解上的差別等等。
(6)頭文件(.h)能夠被頭文件或C文件包含
重複包含(重複定義)
因爲頭文件包含能夠嵌套,那麼C文件就有可能包含屢次同一個頭文件,就可能出現重複定義的問題的。
經過條件編譯開關來避免重複包含(重複定義)
例如
#ifndef __headerfileXXX__
#define __headerfileXXX__
…
文件內容
…
#endif
此功能也可使用#pargam once 指令代替
(7)define中的三個特殊符號:#,##,#@
===========================================================
1. #define Conn(x,y) x##y
2. #define ToChar(x) #@x
3. #define ToString(x) #x
(1)x##y表示什麼?表示x鏈接y,舉例說:
1. int n = Conn(123,456); /* 結果就是n=123456;*/
2. char* str = Conn("asdf", "adf"); /*結果就是 str = "asdfadf";*/
(2)再來看#@x ,其實就是給x加上單引號,結果返回是一個const char。舉例說:
char a = ToChar(1);結果就是a='1';
作個越界試驗char a = ToChar(123);結果就錯了;
可是若是你的參數超過四個字符,編譯器就給給你報錯了!
error C2015: too many characters in constant :P
(3)最後看看#x,估計你也明白了,他是給x加雙引號
char* str = ToString(123132);就成了str="123132";
宏與函數的選擇:
宏與函數間的選擇其實是時間與空間的權衡,宏產生內聯代碼 也就是說在程序中產生語句,若是使用了宏20次 則會把20行代碼插入程序中,若是使用函數20次,那麼程序中只有一份函數語句的拷貝,所以節省了空間,另外一方面,程序的控制必須轉移到函數中並隨後返回調用程序,所以這比內聯代碼花費的時間多。
宏的一個有點事不檢查其中的變量類型
C 標準的預約義宏
_ _DATE_ _ 進行預處理的日期 (「Mmm dd yyy」形式的字符串文字)
_ _FILE_ _ 表明當前源代碼文件名的字符串文字
_ _LINE_ _ 表明當前源代碼文件中的行號的整數常量
_ _STDC_ _ 設置爲1 時 表示該實現遵循C標準
_ _STDC_HOSTED_ _ 在本機環境設置爲1,不然設置爲0
_ _STDC_VERSION_ _ 在C99是設置爲199901L
_ _TIME_ _ 源文件編譯時間 格式爲"hh:mm:ss"
2.#include
(1)兩種使用形式
#include<stdio.h>
#include "myStuuff.h"
在Unix系統中 尖括號告訴預處理器在一個或多個標準系統目錄中尋找文件, 雙引號告訴預處理器如今當前目錄中尋找文件,而後再標準位置選在文件
(2)使用頭文件
頭文件中最多見的形式包括:
a.明顯常量
b.宏函數
c.函數聲明
d.結構模板定義
e.類型定義