第68節(jié):?jiǎn)纹瑱C(jī)C語(yǔ)言的多文件編程技巧
很多人也把多文件編程稱作模塊化編程,其實(shí)我覺(jué)得叫多文件編程會(huì)更加符合實(shí)際一些。多文件編程有兩個(gè)最大的好處,一個(gè)是給我們的程序增加了目錄,方便我們查找。另外一個(gè)好處是方便移植別人已經(jīng)做好的功能程序模塊,利用這個(gè)特點(diǎn),特別適合團(tuán)隊(duì)一起做大型項(xiàng)目。很多初學(xué)者剛開(kāi)始學(xué)多文件編程時(shí),會(huì)經(jīng)常遇到重復(fù)定義等問(wèn)題,想知道怎么解決這些問(wèn)題嗎?只要按照以下鴻哥教的規(guī)則來(lái)做,這些問(wèn)題就不存在了。
第一個(gè):每個(gè)文件保持成雙成對(duì)出現(xiàn)。每個(gè).c源文件必須有一個(gè).h頭文件跟它對(duì)應(yīng),每個(gè).h頭文件必須有一個(gè).c源文件跟它對(duì)應(yīng)。比如:main.c與main.h,delay.c與 delay.h。
第二個(gè):.c源文件只負(fù)責(zé)函數(shù)的定義和變量的定義,但是不負(fù)責(zé)函數(shù)的聲明和變量的聲明。比如:
unsigned char ucLedStep=0; //這個(gè)是全局變量的定義
void led_flicker() //這個(gè)是函數(shù)的定義
{
//…里面是具體代碼內(nèi)容
}
第三個(gè):.h頭文件只負(fù)責(zé)函數(shù)的聲明和變量的聲明,以及常量和IO口的宏定義,但是不負(fù)責(zé)函數(shù)的定義和變量的定義。比如:
#define const_time_level 200//這個(gè)是常量的宏定義
sbit led_dr=P3^5; //這個(gè)是IO口的宏定義
void led_flicker(); //這個(gè)是函數(shù)的聲明
extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明,不能賦初始值
第四個(gè):每個(gè).h頭文件都必須固定以#ifndef,#define,#endif語(yǔ)句為模板,此模板是用來(lái)避免編譯時(shí)由于重復(fù)包含頭文件里面的內(nèi)容而導(dǎo)致出錯(cuò)。其中標(biāo)志變量_XXX_鴻哥建議用它本身的文件名稱加前后下劃線_。
比如:
#ifndef _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
#define _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
#define const_time_level 200//這個(gè)是常量的宏定義
sbit led_dr=P3^5; //這個(gè)是IO口的宏定義
void led_flicker(); //這個(gè)是函數(shù)的聲明
extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明,不能賦初始值
#endif
第五個(gè):每個(gè).h頭文件里都必須聲明它對(duì)應(yīng)的.c源文件里的所有定義函數(shù)和全局變量,注意:.c源文件里所有的全局變量都要在它所對(duì)應(yīng)的.h頭文件里聲明一次,不僅僅是函數(shù),這個(gè)地方很容易被人忽略。
比如:在led.h頭文件中:
void led_flicker(); //這個(gè)是函數(shù)的聲明,因?yàn)樵谶@個(gè)函數(shù)在led.c文件里定義了。
extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明,不許賦初值
第六個(gè):每個(gè).c源文件里都必須包含兩個(gè)文件,一個(gè)是單片機(jī)的系統(tǒng)頭文件REG52.H,另外一個(gè)是它自己本身的頭文件比如initial.h.剩下其它的頭文件看實(shí)際情況來(lái)決定是否調(diào)用,我們用到了哪些文件的函數(shù),全局變量或者宏定義,就需要調(diào)用對(duì)應(yīng)的頭文件。
比如:在initial.c源文件中:
#include"REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
#include"initial.h"http://必須包含它本身的頭文件
/*注釋:
由于本源文件中用到了led_dr的語(yǔ)句,而led_dr是在led.h文件里宏定義的,所以必須把led.h也包含進(jìn)來(lái)
*/
#include"led.h"http://由于本源文件中用到了led_dr的語(yǔ)句,所以必須把led.h也包含進(jìn)來(lái)
void initial_myself()//這個(gè)是函數(shù)定義
{
led_dr=0;//led_dr是在led文件里定義和聲明的
}
第七個(gè):聲明一個(gè)全局變量必須加extern關(guān)鍵字,同時(shí)千萬(wàn)不能在聲明全局變量的時(shí)候賦初始值,比如:
extern unsigned char ucLedStep=0; //這樣是絕對(duì)錯(cuò)誤的。
extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明,這個(gè)才是正確的
第八個(gè):對(duì)于函數(shù)與全局變量的聲明,編譯器都不分配內(nèi)存空間。對(duì)于函數(shù)與全局變量的定義,編譯器都分配內(nèi)存空間。函數(shù)與全局變量的定義只能在一個(gè).c源文件中出現(xiàn)一次,而函數(shù)與全局變量的聲明可以在多個(gè).h文件中出現(xiàn)。
具體內(nèi)容,請(qǐng)看源代碼講解,本程序例程是直接把前面第四節(jié)一個(gè)源文件更改成多文件編程方式。
(1)硬件平臺(tái):
基于朱兆祺51單片機(jī)學(xué)習(xí)板。把前面第四節(jié)一個(gè)源文件更改成多文件編程方式。
(2)實(shí)現(xiàn)功能:跟前面第四節(jié)的功能一模一樣,讓一個(gè)LED閃爍。
(3)keil多文件編程的截圖預(yù)覽:
(4)整個(gè)源代碼講解工程文件下載:
(5)源代碼講解如下(注意,以下代碼不能直接放到一個(gè)源文件里編譯):
- /*以下是 main.h 的內(nèi)容*/
- /* 注釋一:
- 每個(gè)頭文件都是固定以#ifndef,#define,#endif
- 為模板,其中標(biāo)志變量_XXX_我建議用它本身的文件名稱加前后下劃線_。
- 此標(biāo)志變量名稱是用來(lái)預(yù)防多次包含出錯(cuò)的,詳細(xì)講解請(qǐng)看注釋二。
- 每個(gè)頭文件只做函數(shù)的聲明和變量的聲明,以及常量和IO口的宏定義,不做
- 函數(shù)的定義與變量的定義。
- */
- #ifndef _MAIN_ //標(biāo)志變量_MAIN_是用它本身的文件名稱命名
- #define _MAIN_ //標(biāo)志變量_MAIN_是用它本身的文件名稱命名
- void main();//這個(gè)是函數(shù)的聲明
- #endif
- /* 注釋二:
- 以上語(yǔ)句
- #ifndef
- #define
- 插入其它內(nèi)容...
- #endif
- 類似于把_MAIN_看成是一個(gè)標(biāo)志變量
- if(_MAIN_==0)// 相當(dāng)于#ifndef _MAIN_
- {
- _MAIN_=1;// 相當(dāng)于#define _MAIN_
- 插入其它內(nèi)容...
- } //相當(dāng)于#endif
- 目的是通過(guò)一個(gè)標(biāo)志位變量的賦值,讓編譯器在編譯的時(shí)候,只包含一次此頭文件,避免多次包含出錯(cuò)
- */
- /*------分割線--------------------------------------------------*/
- /*以下是 main.c 的內(nèi)容*/
- /* 注釋一:
- 每個(gè)源文件都必須包含兩個(gè)文件,一個(gè)是單片機(jī)的系統(tǒng)頭文件REG52.H,
- 另外一個(gè)是它自己本身的頭文件main.h.剩下其它的頭文件看實(shí)際情況來(lái)
- 決定是否調(diào)用,我們用到了哪些文件的函數(shù),全局變量或者宏定義,就需要調(diào)用對(duì)應(yīng)的頭文件。
- 每個(gè)源文件只做函數(shù)的定義和變量的定義,不做函數(shù)的聲明和變量的聲明。
- */
- #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
- #include "main.h"http://必須包含它本身的頭文件
- /* 注釋二:
- (1)由于本源文件中調(diào)用initial_myself()和initial_peripheral()函數(shù),而這兩個(gè)函數(shù)
- 都是在initial文件里定義和聲明的,所以必須把initial.h也包含進(jìn)來(lái)。
- (2)由于本源文件中調(diào)用delay_long(100)函數(shù),而這個(gè)函數(shù)
- 是在delay文件里定義和聲明的,所以必須把delay.h也包含進(jìn)來(lái)。
- (2)由于本源文件中調(diào)用led_flicker()函數(shù),而這個(gè)函數(shù)
- 是在led文件里定義和聲明的,所以必須把led.h也包含進(jìn)來(lái)。
- */
- #include "initial.h"http://由于本源文件中用到了initial_myself()和initial_peripheral()函數(shù),所以必須把initial.h也包含進(jìn)來(lái)
- #include "delay.h"http://由于本源文件中用到了delay_long(100)函數(shù),所以必須把delay.h也包含進(jìn)來(lái)
- #include "led.h"http://由于本源文件中用到了led_flicker()函數(shù),所以必須把led.h也包含進(jìn)來(lái)
- void main()//這個(gè)是函數(shù)的定義
- {
- initial_myself();
- delay_long(100);
- initial_peripheral();
- while(1)
- {
- led_flicker();
- }
- }
- /*------分割線--------------------------------------------------*/
- /*以下是 delay.h 的內(nèi)容*/
- #ifndef _DELAY_ //標(biāo)志變量_DELAY_是用它本身的文件名稱命名
- #define _DELAY_ //標(biāo)志變量_DELAY_是用它本身的文件名稱命名
- void delay_long(unsigned int uiDelaylong); //這個(gè)是函數(shù)的聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
- #endif
- /*------分割線--------------------------------------------------*/
- /*以下是 delay.c 的內(nèi)容*/
- #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
- #include "delay.h"http://必須包含它本身的頭文件
- void delay_long(unsigned int uiDelayLong)//這個(gè)是函數(shù)的定義
- {
- unsigned int i; //這個(gè)是局部變量的定義
- unsigned int j; //這個(gè)是局部變量的定義
- for(i=0;i
- {
- for(j=0;j<500;j++)
- {
- ;
- }
- }
- }
- /*------分割線--------------------------------------------------*/
- /*以下是 initial.h 的內(nèi)容*/
- #ifndef _INITIAL_ //標(biāo)志變量_INITIAL_是用它本身的文件名稱命名
- #define _INITIAL_ //標(biāo)志變量_INITIAL_是用它本身的文件名稱命名
- void initial_myself(); //這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
- void initial_peripheral(); //這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
- #endif
- /*------分割線--------------------------------------------------*/
- /*以下是 initial.c 的內(nèi)容*/
- #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
- #include "initial.h"http://必須包含它本身的頭文件
- /* 注釋一:
- 由于本源文件中用到了led_dr的語(yǔ)句,而led_dr是在led文件里宏定義的,所以必須把led.h也包含進(jìn)來(lái)
- */
- #include "led.h"http://由于本源文件中用到了led_dr的語(yǔ)句,所以必須把led.h也包含進(jìn)來(lái)
- void initial_myself()//這個(gè)是函數(shù)定義
- {
- TMOD=0x01;//以下能直接用TMOD,TH0,TL0,EA,ET0,TR0這些寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
- TH0=0xf8;
- TL0=0x2f;
- led_dr=0;//led_dr是在led文件里定義和聲明的
- }
- void initial_peripheral() //這個(gè)是函數(shù)定義
- {
- EA=1;
- ET0=1;
- TR0=1;
- }
- /*------分割線--------------------------------------------------*/
- /*以下是 interrupt.h 的內(nèi)容*/
- #ifndef _INTERRUPT_ //標(biāo)志變量_INTERRUPT_是用它本身的文件名稱命名
- #define _INTERRUPT_ //標(biāo)志變量_INTERRUPT_是用它本身的文件名稱命名
- void T0_time();//這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
- /* 注釋一:
- 聲明一個(gè)外部全局變量必須加extern關(guān)鍵字,同時(shí)千萬(wàn)不能在聲明全局變量的時(shí)候賦初始值,比如:
- extern unsigned int uiTimeCnt=0; 這樣是絕對(duì)錯(cuò)誤的。
- */
- extern unsigned int uiTimeCnt; //這個(gè)是全局變量的聲明,不能賦初始值
- #endif
- /*------分割線--------------------------------------------------*/
- /*以下是 interrupt.c 的內(nèi)容*/
- #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
- #include "interrupt.h"http://必須包含它本身的頭文件
- unsigned int uiTimeCnt=0; //這個(gè)是全局變量的定義,可以賦初值
- void T0_time() interrupt 1//這個(gè)是函數(shù)定義
- {
- TF0=0; //以下能直接用TF0,TR0,TH0,TL0這些寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
- TR0=0;
- if(uiTimeCnt<0xffff)
- {
- uiTimeCnt++;
- }
- TH0=0xf8;
- TL0=0x2f;
- TR0=1;
- }
- /*------分割線--------------------------------------------------*/
- /*以下是 led.h 的內(nèi)容*/
- #ifndef _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
- #define _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
- #define const_time_level 200 //宏定義都放在頭文件里
- /* 注釋一:
- IO口的宏定義也放在頭文件里,
- 如果是PIC單片機(jī),以下IO口定義相當(dāng)于宏定義 #defineled_dr LATBbits.LATB4等語(yǔ)句
- */
- sbit led_dr=P3^5; //如果是PIC單片機(jī),相當(dāng)于宏定義 #defineled_dr LATBbits.LATB4等語(yǔ)句
- void led_flicker(); //這個(gè)是函數(shù)的聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
- /* 注釋三:
- 聲明一個(gè)全局變量必須加extern關(guān)鍵字,同時(shí)千萬(wàn)不能在聲明全局變量的時(shí)候賦初始值,比如:
- extern unsigned char ucLedStep=0; 這樣是絕對(duì)錯(cuò)誤的。
- */
- extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明
- #endif
- /*------分割線--------------------------------------------------*/
- /*以下是 led.c 的內(nèi)容*/
- #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
- #include "led.h"http://必須包含它本身的頭文件
- /* 注釋一:
- 由于本源文件中用到了uiTimeCnt全局變量,而uiTimeCnt是在interrupt文件里聲明和定義的,
- 所以必須把interrupt.h也包含進(jìn)來(lái)
- */
- #include "interrupt.h"http://必須包含它本身的頭文件
- unsigned char ucLedStep=0; //這個(gè)是全局變量的定義,可以賦初值
- void led_flicker() //這個(gè)是函數(shù)的定義
- {
- switch(ucLedStep)
- {
- case 0:
- if(uiTimeCnt>=const_time_level)
- {
- ET0=0;//以下能直接用ET0寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
- uiTimeCnt=0; //uiTimeCnt此變量是在interrupt文件里聲明和定義的,所以必須把interrupt.h也包含進(jìn)來(lái)
- ET0=1;
- led_dr=1;//此IO口定義已經(jīng)在led.h頭文件中定義了
- ucLedStep=1; //切換到下一個(gè)步驟
- }
- break;
- case 1:
- if(uiTimeCnt>=const_time_level)
- {
- ET0=0;
- uiTimeCnt=0;
- ET0=1;
- led_dr=0;
- ucLedStep=0; //返回到上一個(gè)步驟
- }
- break;
- }
- }
- /*------分割線--------------------------------------------------*/
總結(jié)陳詞:
下一節(jié)開(kāi)始講液晶屏顯示方面的內(nèi)容。欲知詳情,請(qǐng)聽(tīng)下回分解----帶字庫(kù)12864液晶屏的常用點(diǎn)陣字體程序。
評(píng)論