當(dāng)前位置:首頁(yè) > 單片機(jī) > 單片機(jī)
[導(dǎo)讀]當(dāng)你在一個(gè)項(xiàng)目小組做一個(gè)相對(duì)較復(fù)雜的工程時(shí),意味著你不再獨(dú)自單干。你需要和你的小組成員分工合作,一起完成項(xiàng)目,這就要求小組成員各自負(fù)責(zé)一部分工程。比如你可能只是負(fù)責(zé)通訊或者顯示這一塊。這個(gè)時(shí)候,你就應(yīng)

當(dāng)你在一個(gè)項(xiàng)目小組做一個(gè)相對(duì)較復(fù)雜的工程時(shí),意味著你不再獨(dú)自單干。你需要和你的小組成員分工合作,一起完成項(xiàng)目,這就要求小組成員各自負(fù)責(zé)一部分工程。比如你可能只是負(fù)責(zé)通訊或者顯示這一塊。這個(gè)時(shí)候,你就應(yīng)該將自己的這一塊程序?qū)懗梢粋€(gè)模塊,單獨(dú)調(diào)試,留出接口供其它模塊調(diào)用。最后,小組成員都將自己負(fù)責(zé)的模塊寫(xiě)完并調(diào)試無(wú)誤后,由項(xiàng)目組長(zhǎng)進(jìn)行組合調(diào)試。像這些場(chǎng)合就要求程序必須模塊化。模塊化的好處是很多的,不僅僅是便于分工,它還有助于程序的調(diào)試,有利于程序結(jié)構(gòu)的劃分,還能增加程序的可讀性和可移植性。
初學(xué)者往往搞不懂如何模塊化編程,其實(shí)它是簡(jiǎn)單易學(xué),而且又是組織良好程序結(jié)構(gòu)行之有效的方法之一.
本文將先大概講一下模塊化的方法和注意事項(xiàng),最后將以初學(xué)者使用最廣的keil c編譯器為例,給出模塊化編程的詳細(xì)步驟。
模塊化程序設(shè)計(jì)應(yīng)該理解以下概述:
(1) 模塊即是一個(gè).c 文件和一個(gè).h 文件的結(jié)合,頭文件(.h)中是對(duì)于該模塊接口的聲明;
這一條概括了模塊化的實(shí)現(xiàn)方法和實(shí)質(zhì):將一個(gè)功能模塊的代碼單獨(dú)編寫(xiě)成一個(gè).c文件,然后把該模塊的接口函數(shù)放在.h文件中.舉例:假如你用到液晶顯示,那么你可能會(huì)寫(xiě)一個(gè)液晶驅(qū)動(dòng)模塊,以實(shí)現(xiàn)字符、漢字和圖像的現(xiàn)實(shí),命名為: led_device.c,該模塊的.c文件大體可以寫(xiě)成:

#include …

//定義變量
unsigned char value;//全局變量

//定義函數(shù)
//這是本模塊第一個(gè)函數(shù),起到延時(shí)作用,只供本模塊的函數(shù)調(diào)用,所以用到static關(guān)鍵字修飾

static void delay (uint us) //delay time
{}
//這是本模塊的第二個(gè)函數(shù),要在其他模塊中調(diào)用

void wr_lcd (uchar dat_comm,uchar content)
{}
……
……

注:此處只寫(xiě)出這兩個(gè)函數(shù),第一個(gè)延時(shí)函數(shù)的作用范圍是模塊內(nèi),第二個(gè),它是其它模塊需要的。為了簡(jiǎn)化,此處并沒(méi)有寫(xiě)出函數(shù)體.

.h文件中給出模塊的接口.在上面的例子中,向LCD寫(xiě)入字符函數(shù):wr_lcd (uchar dat_comm,uchar content)就是一個(gè)接口函數(shù),因?yàn)槠渌K會(huì)調(diào)用它,那么.h文件中就必須將這個(gè)函數(shù)聲明為外部函數(shù)(使用extrun關(guān)鍵字修飾),另一個(gè)延時(shí)函數(shù):void delay (uint us)只是在本模塊中使用(本地函數(shù),用static關(guān)鍵字修飾),因此它是不需要放到.h文件中的。
.h文件格式如下:

//聲明全局變量
extern unsigned char value;
//聲明接口函數(shù)
extern void wr_lcd (uchar dat_comm,uchar content); //向LCD寫(xiě)入字符
……

這里注意三點(diǎn):
1. 在keil 編譯器中,extern這個(gè)關(guān)鍵字即使不聲明,編譯器也不會(huì)報(bào)錯(cuò),且程序運(yùn)行良好,但不保證使用其它編譯器也如此。強(qiáng)烈建議加上,養(yǎng)成良好的編程規(guī)范。
2. .c文件中的函數(shù)只有其它模塊使用時(shí)才會(huì)出現(xiàn)在.h文件中,像本地延時(shí)函數(shù)static void delay (uint us)即使出現(xiàn)在.h文件中也是在做無(wú)用功,因?yàn)槠渌K根本不去調(diào)用它,實(shí)際上也調(diào)用不了它(static關(guān)鍵字的限制作用)。
3.注意本句最后一定要加分號(hào)”;”,相信有不少同學(xué)遇到過(guò)這個(gè)奇怪的編譯器報(bào)錯(cuò): error C132: 'xxxx': not in formal parameter list,這個(gè)錯(cuò)誤其實(shí)是.h的函數(shù)聲明的最后少了分號(hào)的緣故。

模塊的應(yīng)用:假如需要在LCD菜單模塊lcd_menu.c中使用液晶驅(qū)動(dòng)模塊lcd_device.c中的函數(shù)void wr_lcd (uchar dat_comm,uchar content),只需在LCD菜單模塊的lcd_menu.c文件中加入液晶驅(qū)動(dòng)模塊的頭文件lcd_device.h即可.

#include“l(fā)cd_device.h //包含液晶驅(qū)動(dòng)程序頭文件,之后就可以在該.c文件中調(diào)用//lcd_device.h中的全局函數(shù),使用液晶驅(qū)動(dòng)程序里的全局//變量(如果有的話)。

//調(diào)用向LCD寫(xiě)入字符函數(shù)
wr_lcd (0x01,0x30);

//對(duì)全局變量賦值
value=0xff;

(2) 某模塊提供給其它模塊調(diào)用的外部函數(shù)及數(shù)據(jù)需在.h 中文件中冠以extern 關(guān)鍵字聲明;
這句話在上面的例子中已經(jīng)有體現(xiàn),即某模塊提供給其它模塊調(diào)用的外部函數(shù)和全局變量需在.h 中文件中冠以extern 關(guān)鍵字聲明,下面重點(diǎn)說(shuō)一下全局變量的使用。使用模塊化編程的一個(gè)難點(diǎn)(相對(duì)于新手)就是全局變量的設(shè)定,初學(xué)者往往很難想通模塊與模塊公用的變量是如何實(shí)現(xiàn)的,常規(guī)的做法就是本句提到的,在.h文件中外部數(shù)據(jù)冠以extern關(guān)鍵字聲明。比如上例的變量value就是一個(gè)全局變量,若是某個(gè)模塊也使用這個(gè)變量,則和使用外部函數(shù)一樣,只需在使用的模塊.c文件中包含#include“l(fā)cd_device.h”即可。
另一種處理模塊間全局變量的方法來(lái)自于嵌入式操作系統(tǒng)uCOS-II,這個(gè)操作系統(tǒng)處理全局變量的方法比較特殊,也比較難以理解,但學(xué)會(huì)之后妙用無(wú)窮,這個(gè)方法只需用在頭文件中定義一次。方法為:
在定義所有全局變量(uCOS-II將所有全局變量定義在一個(gè).h文件內(nèi))的.h頭文件中:
#ifdef xxx_GLOBALS
#define xxx_EXT
#else
#define xxx_EXT extern
#endif
.H 文件中每個(gè)全局變量都加上了xxx_EXT的前綴。xxx 代表模塊的名字。
該模塊的.C文件中有以下定義:
#define xxx_GLOBALS
#include "includes.h"
當(dāng)編譯器處理.C文件時(shí),它強(qiáng)制xxx_EXT(在相應(yīng).H文件中可以找到)為空,(因?yàn)閤xx_GLOBALS已經(jīng)定義)。所以編譯器給每個(gè)全局變量分配內(nèi)存空間,而當(dāng)編譯器處理其他.C 文件時(shí),xxx_GLOBAL沒(méi)有定義,xxx_EXT 被定義為extern,這樣用戶(hù)就可以調(diào)用外部全局變量。為了說(shuō)明這個(gè)概念,可以參見(jiàn)uC/OS_II.H,其中包括以下定義:
#ifdef OS_GLOBALS
#define OS_EXT
#else
#define OS_EXT extern
#endif
OS_EXT INT32U OSIdleCtr;
OS_EXT INT32U OSIdleCtrRun;
OS_EXT INT32U OSIdleCtrMax;
同時(shí),uCOS_II.H 有中以下定義:
#define OS_GLOBALS
#include “includes.h”
當(dāng)編譯器處理uCOS_II.C 時(shí),它使得頭文件變成如下所示,因?yàn)镺S_EXT 被設(shè)置為空。
INT32U OSIdleCtr;
INT32U OSIdleCtrRun;
INT32U OSIdleCtrMax;
這樣編譯器就會(huì)將這些全局變量分配在內(nèi)存中。當(dāng)編譯器處理其他.C 文件時(shí),頭文件變成了如下的樣子,因?yàn)镺S_GLOBAL沒(méi)有定義,所以O(shè)S_EXT 被定義為extern。
extern INT32U OSIdleCtr;
extern INT32U OSIdleCtrRun;
extern INT32U OSIdleCtrMax;
在這種情況下,不產(chǎn)生內(nèi)存分配,而任何 .C文件都可以使用這些變量。這樣的就只需在 .H文件中定義一次就可以了。
(3) 模塊內(nèi)的函數(shù)和全局變量需在.c 文件開(kāi)頭冠以static 關(guān)鍵字聲明;
這句話主要講述了關(guān)鍵字static的作用。Static是一個(gè)相當(dāng)重要的關(guān)鍵字,他能對(duì)函數(shù)和變量做一些約束,而且可以傳遞一些信息。比如上例在LCD驅(qū)動(dòng)模塊.c文件中定義的延時(shí)函數(shù)static void delay (uint us),這個(gè)函數(shù)冠以static修飾,一方面是限定了函數(shù)的作用范圍只是在本模塊中起作用,另一方面也給人傳達(dá)這樣的信息:該函數(shù)不會(huì)被其他模塊調(diào)用。下面詳細(xì)說(shuō)一下這個(gè)關(guān)鍵字的作用,在C 語(yǔ)言中,關(guān)鍵字static 有三個(gè)明顯的作用:
1.在函數(shù)體,一個(gè)被聲明為靜態(tài)的變量在這一函數(shù)被調(diào)用過(guò)程中維持其值不變。
2.在模塊內(nèi)(但在函數(shù)體外),一個(gè)被聲明為靜態(tài)的變量可以被模塊內(nèi)所用函數(shù)訪問(wèn),但不能被模塊外其它函數(shù)訪問(wèn)。它是一個(gè)本地的全局變
量。
3.在模塊內(nèi),一個(gè)被聲明為靜態(tài)的函數(shù)只可被這一模塊內(nèi)的其它函數(shù)調(diào)用。那就是,這個(gè)函數(shù)被限制在聲明它的模塊的本地范圍內(nèi)使用。
前兩個(gè)都比較容易理解,最后一個(gè)作用就是剛剛舉例中提到的延時(shí)函數(shù)(static void delay (uint us)),本地化函數(shù)是有相當(dāng)好的作用的。
(4) 永遠(yuǎn)不要在.h 文件中定義變量!
呵呵,似乎有點(diǎn)危言聳聽(tīng)的感覺(jué),但我想也不會(huì)有多少人會(huì)在.h文件中定義變量的。
比較一下代碼:
代碼一:

int a = 5;

#include "module1.h"


#include "module1.h"


#include "module1.h"
以上程序的結(jié)果是在模塊1、2、3 中都定義了整型變量a,a 在不同的模塊中對(duì)應(yīng)不同的地址元,這個(gè)世界上從來(lái)不需要這樣的程序。正確的做法是:
代碼二:

extern int a;


#include "module1.h"
int a = 5;


#include "module1.h"


#include "module1.h"
這樣如果模塊1、2、3 操作a 的話,對(duì)應(yīng)的是同一片內(nèi)存單元。
注:
一個(gè)嵌入式系統(tǒng)通常包括兩類(lèi)(注意是兩類(lèi),不是兩個(gè))模塊:
(1)硬件驅(qū)動(dòng)模塊,一種特定硬件對(duì)應(yīng)一個(gè)模塊;
(2)軟件功能模塊,其模塊的劃分應(yīng)滿(mǎn)足低偶合、高內(nèi)聚的要求。
下面以keil C 編譯器為例,講一下模塊化編程的步驟。
下面這個(gè)程序分為三層,共7個(gè)模塊,共同為主程序服務(wù)(它們之間也會(huì)相互調(diào)用)。
程序的結(jié)構(gòu)圖如下所示:


程序主要模塊和功能簡(jiǎn)介:
一. 底層驅(qū)動(dòng)
1. 紅外鍵盤(pán):程序通過(guò)紅外鍵盤(pán)進(jìn)行操作。紅外鍵盤(pán)獨(dú)占定時(shí)器0和外部中斷0,以實(shí)現(xiàn)紅外解碼和鍵盤(pán)鍵值的識(shí)別。紅外鍵盤(pán)定義了五個(gè)按鍵,分別為上翻、下翻、左翻、右翻和確認(rèn)鍵。
2. LCD液晶顯示:程序主要通過(guò)LCD顯示信息,LCD液晶顯示驅(qū)動(dòng)提供顯示漢字、圖形和ASCII碼的函數(shù)接口??梢匀?、單行顯示漢字,任意位置顯示ASCII碼,還可以全屏、半屏顯示圖形。
二. 功能模塊
1. LCD菜單程序:菜單程序可以使人機(jī)交互更加方便、容易。本菜單程序的菜單級(jí)別深度受RAM大小的限制,每增加一級(jí)菜單將多消耗4字節(jié)的RAM。菜單程序主要完成菜單功能函數(shù)的調(diào)度,LCD顯示刷新。
2. 計(jì)算器程序:實(shí)現(xiàn)65536以?xún)?nèi)的加、減、乘、除,超出范圍會(huì)出現(xiàn)溢出,溢出發(fā)生時(shí),LCD顯示“錯(cuò)誤:出現(xiàn)溢出”的錯(cuò)誤提示,同時(shí)本次運(yùn)算被忽略。對(duì)于負(fù)數(shù)會(huì)顯示“-”號(hào),除數(shù)為零時(shí)LCD顯示“錯(cuò)誤:除數(shù)為零”的錯(cuò)誤提示。
3. 開(kāi)機(jī)次數(shù)記憶程序:主要對(duì)基于IIC總線的EEPROM進(jìn)行讀寫(xiě),

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請(qǐng)聯(lián)系該專(zhuān)欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請(qǐng)及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車(chē)的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

倫敦2024年8月29日 /美通社/ -- 英國(guó)汽車(chē)技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車(chē)工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車(chē)。 SODA V工具的開(kāi)發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車(chē) 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來(lái)越多用戶(hù)希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來(lái)越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對(duì)日本游戲市場(chǎng)的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)開(kāi)幕式在貴陽(yáng)舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國(guó)國(guó)際大數(shù)據(jù)產(chǎn)業(yè)博覽會(huì)上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱(chēng),數(shù)字世界的話語(yǔ)權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對(duì)環(huán)境變化,經(jīng)營(yíng)業(yè)績(jī)穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤(rùn)率延續(xù)升勢(shì) 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長(zhǎng) 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競(jìng)爭(zhēng)力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競(jìng)爭(zhēng)優(yōu)勢(shì)...

關(guān)鍵字: 通信 BSP 電信運(yùn)營(yíng)商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺(tái)與中國(guó)電影電視技術(shù)學(xué)會(huì)聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會(huì)上宣布正式成立。 活動(dòng)現(xiàn)場(chǎng) NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長(zhǎng)三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會(huì)上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡(jiǎn)稱(chēng)"軟通動(dòng)力")與長(zhǎng)三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉