對於一個 C 程式 編譯之前, C 編譯器 會先處理 C 程式 中含有
#define、#undef、#if、#ifdef、#ifndef、#endif、#elif、#else 及 #include 敘述, 將之置換成一新 C 程式, 再將之編譯 成一 assembly 程式, 再轉成 machine code(或 object code)。 再經由 linker 變成一執行檔。
#define 巨集的宣告與使用
#define macroname string其說明如下:
- macroname 為一連串的字元(不含空白字元) 或是類似函數的標題,如 macroname(arg1, arg2) 含有左括號及右括號。
- 編譯器於前置處理時會將程式中出現 macroname 以 string 來替代。
- 巨集指令基本上是以一行為單位。因此替代字串是從 macroname 後第一個非空白字元起至該行結束。
- 如果巨集指令超過一行,則以反斜線 \ 表示下一行繼續下去。
巨集指令的使用(或呼叫),如同函數的呼叫,將 macroname 置於程式中, 為一辨識個體(identifier)即可。對於含引數型的巨集指令,其呼叫方式與函數的呼叫型式完全一樣。
巨集指令的使用範圍為有宣告處起至檔案的結束,一般而言,巨集指令 都是宣告於檔案的開端,或是宣告於檔頭,即檔案其副檔名為 .h 再以 #include 的方式加入。
=====================================
例 1: 設有一 C 程式檔 file.c 其內容為:
=====================================
====================================
為了程式易讀易維修,程式中的常數通常都是以巨集方式來宣告一常數 名稱,該名稱通常是以大寫的字串來表示。
====================================
例 2: 設有一 C 程式其內容如下:
====================================
含引數型的巨集指令,其宣告格式為
========================================
例 3:對於巨集指令的宣告
=========================================
例 4:對於巨集指令的宣告
以下是將範例Blink以巨集指令宣告的方式修改得來
#include 的使用
C 程式的設計,不僅可以用模組化的方式,以函數來表現,亦可將程式碼 依其不同的目的與功能,呈現在不同的檔案中。為了程式碼的一致性,如 常數,巨集指令,全域(global)變數的宣告(含 extern)一 些函數的宣告(僅列出函數的標題 heading)等,都放在一檔頭檔 .h, 當然不見的是要以 .h 當作副檔名,不過,一般都是以 .h 來當作 include 檔的副檔名, 再以指令 #include 方式將該檔 加入即可。指令 #include 的宣告方式如下
================================================
經過編譯器前置處理後,含指令 #include 這行則由該檔 filename 來代替,
如果 filename 是由 <及> 所含,則編譯器會到 include 目錄下去找該檔或至
預先所定的目錄去找。
=================================================
例 6: 假設檔案 file1.h 的內容如下:
條件編譯: #if…#elif…#else…#endif , #ifdef…#else…#endif , #ifndef…#else…#endif
如果我們在一檔案欲 include 一檔頭檔 file1.h ,又想避免在同一檔案宣告兩次相 同的東西,例如:
欲避免這種情況發生,每個 #include "file1.h" 指令, 我們可以改變為如下指令:
巨集指令的使用範圍為有宣告處起至檔案的結束,一般而言,巨集指令 都是宣告於檔案的開端,或是宣告於檔頭,即檔案其副檔名為 .h 再以 #include 的方式加入。
=====================================
例 1: 設有一 C 程式檔 file.c 其內容為:
=====================================
#define MAXSIZE 100 void setup( ) { int A[MAXSIZE]; }
==============================================於編譯時經過前置處理後,該程式變成
====================================
void setup( ) { int A[100]; }
==============================================其中所有的 #define 指令都不見了,MAXSIZE 也都變成了 100。當然前 置處理後的結果我們也見不到。
為了程式易讀易維修,程式中的常數通常都是以巨集方式來宣告一常數 名稱,該名稱通常是以大寫的字串來表示。
====================================
例 2: 設有一 C 程式其內容如下:
====================================
#define MAXSIZE 100 : : main( ) { int A[MAXSIZE]; .....;} void f( ) { ...... for (i=0;i < MAXSIZE; i++) ......... }
==============================================說明:
- 由辨識體 MAXSIZE 本身即可明白該文的意義。
- 如果陣列的大小要改為 200,只要改巨集指令為 #define MAXSIZE 200 即可, 整個程式仍保持其一致性。
含引數型的巨集指令,其宣告格式為
#define macroname(arg1[,arg]) string1 arg1其說明如下:
- macroname 為該巨集指令的名稱
- arg1 為其引數
- string1 arg1 為其替代字串
========================================
例 3:對於巨集指令的宣告
#define set_nonnegative(A); if (A<0)\ A=A*(-1);
==================================================則於程式中
set_nonnegative(i);
==================================================經過前置處理後即變成
if(i<0) i=i*(-1);其中形式引數 A 皆由實際引數 i 來替代。
=========================================
例 4:對於巨集指令的宣告
#define swap_int(A,B); {int t;\ t=A;A=B;B=t;}於程式中巨集呼叫
swap_int(i,j);經過前置處理後即變成
{int t; t=i; i=j; j=t;}
===================================================
含引數的巨集指令,稍一不慎就會產生錯誤, 如例 4 中 swap_int(s,t); 則會產生問題, 為什麼? =================================================== 巨集指令的宣告亦有階層式,即可利用前面宣告過的巨集指令,如例 5。 例 5:階層式巨集指令========================================#define max(A,B) ((A>B)?A:B) #define max3(A,B,C) max( max(A,B),c)===================================================問題:試比較 macro(巨集) 與 function(函數) 的優劣點。
C++ 的 inline 作用如同 macro 的作用。
以下是將範例Blink以巨集指令宣告的方式修改得來
#define LED1 13 #define SET_outline(A) pinMode(A,OUTPUT); #define SET_nonnegtaive1(A) digitalWrite(A,HIGH); #define SET_nonnegtaive2(A) digitalWrite(A,LOW); void setup() { SET_outline(LED1); } void loop() { SET_nonnegtaive1(LED1); delay(500); SET_nonnegtaive2(LED1); delay(500); }=========================================
#include 的使用
C 程式的設計,不僅可以用模組化的方式,以函數來表現,亦可將程式碼 依其不同的目的與功能,呈現在不同的檔案中。為了程式碼的一致性,如 常數,巨集指令,全域(global)變數的宣告(含 extern)一 些函數的宣告(僅列出函數的標題 heading)等,都放在一檔頭檔 .h, 當然不見的是要以 .h 當作副檔名,不過,一般都是以 .h 來當作 include 檔的副檔名, 再以指令 #include 方式將該檔 加入即可。指令 #include 的宣告方式如下
================================================
#include <filename>或
#include "filename"================================================
經過編譯器前置處理後,含指令 #include 這行則由該檔 filename 來代替,
如果 filename 是由 <及> 所含,則編譯器會到 include 目錄下去找該檔或至
預先所定的目錄去找。
=================================================
例 6: 假設檔案 file1.h 的內容如下:
#define MAXSIZE 100 extern int size; int f(int x);假設檔案 file2.c 的內容如下:
#include "file1.h" main() { int A[MAXSIZE],i,sum; for(i=0;i < size;i++) sum=sum+f(i); }又假設 file1.h 與 file2.c 都在同一目錄下 則經過前置處理, file2.c 就先變成
#define MAXSIZE 100 extern int size; int f(int x); main() { int A[MAXSIZE],i,sum; for(i=0;i < size;i++) sum=sum+f(i); }經過前置處理完成後,file2.c 就變成
extern int size; int f(int x); main() { int A[100],i,sum; for(i=0;i < size;i++) sum=sum+f(x); }註:
- 檔頭檔(.h 檔)亦可含 #include 指令而形成階層式的架構。
- 一個檔案可能會含有相當多 #include 的指令,有時會有重複重宣告情形而產 生錯誤。
條件編譯: #if…#elif…#else…#endif , #ifdef…#else…#endif , #ifndef…#else…#endif
如果我們在一檔案欲 include 一檔頭檔 file1.h ,又想避免在同一檔案宣告兩次相 同的東西,例如:
- 檔案 file1.h 含 extern int size
- 檔案 file2.h 含 #include "file1.h"
- 檔案 file3.h 含 #include "file1.h"
- 檔案 file.c 含 #include "file2.h" 及 #include "file3.h"
欲避免這種情況發生,每個 #include "file1.h" 指令, 我們可以改變為如下指令:
#if !define(FILE1) //Arduino 需用 #ifndef #define FILE1 "file1.h" #include FILE1 #endif或是
#ifndef FILE1 #define FILE1 "file1.h" #include FILE1 #endif註:
- 如果一個辨識名稱 X 沒有被 define 過, 則 define(X) 之值為 0, 或者 ifndef X 為真值。
- 當 file2.h 與 file3.h 都含上述指令, 如此一來 file1.h 僅能一次被加入 file.c 而不致於產生衝突。
沒有留言:
張貼留言