新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 第68節(jié):?jiǎn)纹瑱C(jī)C語(yǔ)言的多文件編程技巧

第68節(jié):?jiǎn)纹瑱C(jī)C語(yǔ)言的多文件編程技巧

作者: 時(shí)間:2016-11-22 來(lái)源:網(wǎng)絡(luò) 收藏
開(kāi)場(chǎng)白:
很多人也把多文件編程稱作模塊化編程,其實(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è)源代碼講解工程文件下載:
more_file_4.rar(23.69 KB, 下載次數(shù): 214)
(5)源代碼講解如下(注意,以下代碼不能直接放到一個(gè)源文件里編譯):
  1. /*以下是 main.h 的內(nèi)容*/
  2. /* 注釋一:
  3. 每個(gè)頭文件都是固定以#ifndef,#define,#endif
  4. 為模板,其中標(biāo)志變量_XXX_我建議用它本身的文件名稱加前后下劃線_。
  5. 此標(biāo)志變量名稱是用來(lái)預(yù)防多次包含出錯(cuò)的,詳細(xì)講解請(qǐng)看注釋二。
  6. 每個(gè)頭文件只做函數(shù)的聲明和變量的聲明,以及常量和IO口的宏定義,不做
  7. 函數(shù)的定義與變量的定義。
  8. */
  9. #ifndef _MAIN_ //標(biāo)志變量_MAIN_是用它本身的文件名稱命名
  10. #define _MAIN_ //標(biāo)志變量_MAIN_是用它本身的文件名稱命名
  11. void main();//這個(gè)是函數(shù)的聲明
  12. #endif
  13. /* 注釋二:
  14. 以上語(yǔ)句
  15. #ifndef
  16. #define
  17. 插入其它內(nèi)容...
  18. #endif
  19. 類似于把_MAIN_看成是一個(gè)標(biāo)志變量
  20. if(_MAIN_==0)// 相當(dāng)于#ifndef _MAIN_
  21. {
  22. _MAIN_=1;// 相當(dāng)于#define _MAIN_
  23. 插入其它內(nèi)容...
  24. } //相當(dāng)于#endif
  25. 目的是通過(guò)一個(gè)標(biāo)志位變量的賦值,讓編譯器在編譯的時(shí)候,只包含一次此頭文件,避免多次包含出錯(cuò)
  26. */
  27. /*------分割線--------------------------------------------------*/
  28. /*以下是 main.c 的內(nèi)容*/
  29. /* 注釋一:
  30. 每個(gè)源文件都必須包含兩個(gè)文件,一個(gè)是單片機(jī)的系統(tǒng)頭文件REG52.H,
  31. 另外一個(gè)是它自己本身的頭文件main.h.剩下其它的頭文件看實(shí)際情況來(lái)
  32. 決定是否調(diào)用,我們用到了哪些文件的函數(shù),全局變量或者宏定義,就需要調(diào)用對(duì)應(yīng)的頭文件。
  33. 每個(gè)源文件只做函數(shù)的定義和變量的定義,不做函數(shù)的聲明和變量的聲明。
  34. */
  35. #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
  36. #include "main.h"http://必須包含它本身的頭文件
  37. /* 注釋二:
  38. (1)由于本源文件中調(diào)用initial_myself()和initial_peripheral()函數(shù),而這兩個(gè)函數(shù)
  39. 都是在initial文件里定義和聲明的,所以必須把initial.h也包含進(jìn)來(lái)。
  40. (2)由于本源文件中調(diào)用delay_long(100)函數(shù),而這個(gè)函數(shù)
  41. 是在delay文件里定義和聲明的,所以必須把delay.h也包含進(jìn)來(lái)。
  42. (2)由于本源文件中調(diào)用led_flicker()函數(shù),而這個(gè)函數(shù)
  43. 是在led文件里定義和聲明的,所以必須把led.h也包含進(jìn)來(lái)。
  44. */
  45. #include "initial.h"http://由于本源文件中用到了initial_myself()和initial_peripheral()函數(shù),所以必須把initial.h也包含進(jìn)來(lái)
  46. #include "delay.h"http://由于本源文件中用到了delay_long(100)函數(shù),所以必須把delay.h也包含進(jìn)來(lái)
  47. #include "led.h"http://由于本源文件中用到了led_flicker()函數(shù),所以必須把led.h也包含進(jìn)來(lái)
  48. void main()//這個(gè)是函數(shù)的定義
  49. {
  50. initial_myself();
  51. delay_long(100);
  52. initial_peripheral();
  53. while(1)
  54. {
  55. led_flicker();
  56. }
  57. }
  58. /*------分割線--------------------------------------------------*/
  59. /*以下是 delay.h 的內(nèi)容*/
  60. #ifndef _DELAY_ //標(biāo)志變量_DELAY_是用它本身的文件名稱命名
  61. #define _DELAY_ //標(biāo)志變量_DELAY_是用它本身的文件名稱命名
  62. void delay_long(unsigned int uiDelaylong); //這個(gè)是函數(shù)的聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
  63. #endif
  64. /*------分割線--------------------------------------------------*/
  65. /*以下是 delay.c 的內(nèi)容*/
  66. #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
  67. #include "delay.h"http://必須包含它本身的頭文件
  68. void delay_long(unsigned int uiDelayLong)//這個(gè)是函數(shù)的定義
  69. {
  70. unsigned int i; //這個(gè)是局部變量的定義
  71. unsigned int j; //這個(gè)是局部變量的定義
  72. for(i=0;i
  73. {
  74. for(j=0;j<500;j++)
  75. {
  76. ;
  77. }
  78. }
  79. }
  80. /*------分割線--------------------------------------------------*/
  81. /*以下是 initial.h 的內(nèi)容*/
  82. #ifndef _INITIAL_ //標(biāo)志變量_INITIAL_是用它本身的文件名稱命名
  83. #define _INITIAL_ //標(biāo)志變量_INITIAL_是用它本身的文件名稱命名
  84. void initial_myself(); //這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
  85. void initial_peripheral(); //這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
  86. #endif
  87. /*------分割線--------------------------------------------------*/
  88. /*以下是 initial.c 的內(nèi)容*/
  89. #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
  90. #include "initial.h"http://必須包含它本身的頭文件
  91. /* 注釋一:
  92. 由于本源文件中用到了led_dr的語(yǔ)句,而led_dr是在led文件里宏定義的,所以必須把led.h也包含進(jìn)來(lái)
  93. */
  94. #include "led.h"http://由于本源文件中用到了led_dr的語(yǔ)句,所以必須把led.h也包含進(jìn)來(lái)
  95. void initial_myself()//這個(gè)是函數(shù)定義
  96. {
  97. TMOD=0x01;//以下能直接用TMOD,TH0,TL0,EA,ET0,TR0這些寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
  98. TH0=0xf8;
  99. TL0=0x2f;
  100. led_dr=0;//led_dr是在led文件里定義和聲明的
  101. }
  102. void initial_peripheral() //這個(gè)是函數(shù)定義
  103. {
  104. EA=1;
  105. ET0=1;
  106. TR0=1;
  107. }
  108. /*------分割線--------------------------------------------------*/
  109. /*以下是 interrupt.h 的內(nèi)容*/
  110. #ifndef _INTERRUPT_ //標(biāo)志變量_INTERRUPT_是用它本身的文件名稱命名
  111. #define _INTERRUPT_ //標(biāo)志變量_INTERRUPT_是用它本身的文件名稱命名
  112. void T0_time();//這個(gè)是函數(shù)聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
  113. /* 注釋一:
  114. 聲明一個(gè)外部全局變量必須加extern關(guān)鍵字,同時(shí)千萬(wàn)不能在聲明全局變量的時(shí)候賦初始值,比如:
  115. extern unsigned int uiTimeCnt=0; 這樣是絕對(duì)錯(cuò)誤的。
  116. */
  117. extern unsigned int uiTimeCnt; //這個(gè)是全局變量的聲明,不能賦初始值
  118. #endif
  119. /*------分割線--------------------------------------------------*/
  120. /*以下是 interrupt.c 的內(nèi)容*/
  121. #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
  122. #include "interrupt.h"http://必須包含它本身的頭文件
  123. unsigned int uiTimeCnt=0; //這個(gè)是全局變量的定義,可以賦初值
  124. void T0_time() interrupt 1//這個(gè)是函數(shù)定義
  125. {
  126. TF0=0; //以下能直接用TF0,TR0,TH0,TL0這些寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
  127. TR0=0;
  128. if(uiTimeCnt<0xffff)
  129. {
  130. uiTimeCnt++;
  131. }
  132. TH0=0xf8;
  133. TL0=0x2f;
  134. TR0=1;
  135. }
  136. /*------分割線--------------------------------------------------*/
  137. /*以下是 led.h 的內(nèi)容*/
  138. #ifndef _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
  139. #define _LED_ //標(biāo)志變量_LED_是用它本身的文件名稱命名
  140. #define const_time_level 200 //宏定義都放在頭文件里
  141. /* 注釋一:
  142. IO口的宏定義也放在頭文件里,
  143. 如果是PIC單片機(jī),以下IO口定義相當(dāng)于宏定義 #defineled_dr LATBbits.LATB4等語(yǔ)句
  144. */
  145. sbit led_dr=P3^5; //如果是PIC單片機(jī),相當(dāng)于宏定義 #defineled_dr LATBbits.LATB4等語(yǔ)句
  146. void led_flicker(); //這個(gè)是函數(shù)的聲明,每一個(gè)源文件里的函數(shù)都要在它的頭文件里聲明
  147. /* 注釋三:
  148. 聲明一個(gè)全局變量必須加extern關(guān)鍵字,同時(shí)千萬(wàn)不能在聲明全局變量的時(shí)候賦初始值,比如:
  149. extern unsigned char ucLedStep=0; 這樣是絕對(duì)錯(cuò)誤的。
  150. */
  151. extern unsigned char ucLedStep; //這個(gè)是全局變量的聲明
  152. #endif
  153. /*------分割線--------------------------------------------------*/
  154. /*以下是 led.c 的內(nèi)容*/
  155. #include "REG52.H"http://必須包含的單片機(jī)系統(tǒng)頭文件
  156. #include "led.h"http://必須包含它本身的頭文件
  157. /* 注釋一:
  158. 由于本源文件中用到了uiTimeCnt全局變量,而uiTimeCnt是在interrupt文件里聲明和定義的,
  159. 所以必須把interrupt.h也包含進(jìn)來(lái)
  160. */
  161. #include "interrupt.h"http://必須包含它本身的頭文件
  162. unsigned char ucLedStep=0; //這個(gè)是全局變量的定義,可以賦初值
  163. void led_flicker() //這個(gè)是函數(shù)的定義
  164. {
  165. switch(ucLedStep)
  166. {
  167. case 0:
  168. if(uiTimeCnt>=const_time_level)
  169. {
  170. ET0=0;//以下能直接用ET0寄存器關(guān)鍵字,是因?yàn)榘薘EG52.H頭文件
  171. uiTimeCnt=0; //uiTimeCnt此變量是在interrupt文件里聲明和定義的,所以必須把interrupt.h也包含進(jìn)來(lái)
  172. ET0=1;
  173. led_dr=1;//此IO口定義已經(jīng)在led.h頭文件中定義了
  174. ucLedStep=1; //切換到下一個(gè)步驟
  175. }
  176. break;
  177. case 1:
  178. if(uiTimeCnt>=const_time_level)
  179. {
  180. ET0=0;
  181. uiTimeCnt=0;
  182. ET0=1;
  183. led_dr=0;
  184. ucLedStep=0; //返回到上一個(gè)步驟
  185. }
  186. break;
  187. }
  188. }
  189. /*------分割線--------------------------------------------------*/
總結(jié)陳詞:
下一節(jié)開(kāi)始講液晶屏顯示方面的內(nèi)容。欲知詳情,請(qǐng)聽(tīng)下回分解----帶字庫(kù)12864液晶屏的常用點(diǎn)陣字體程序。


評(píng)論


技術(shù)專區(qū)

關(guān)閉