新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > 8位單片機(jī)的程序優(yōu)化11條

8位單片機(jī)的程序優(yōu)化11條

作者: 時(shí)間:2016-11-23 來源:網(wǎng)絡(luò) 收藏
1 采用短變量
一個(gè)提高代碼效率的最基本的方式就是減小變量的長度。使用 C 編程時(shí),我們都習(xí)慣于對(duì)循環(huán)控制變量使用 int 類型,這對(duì) 8 位的單片機(jī)來說是一種極大的浪費(fèi),你應(yīng)該仔細(xì)考慮你所聲明的變量值可能的范圍,然后選擇合適的變量類型,很明顯,經(jīng)常使用的變量應(yīng)該是unsigned char,只占用一個(gè)字節(jié)。
2 使用無符號(hào)類型
為什么要使用無符號(hào)類型呢?原因是8051不支持符號(hào)運(yùn)算,程序中也不要使用含有帶符號(hào)變量的外部代碼,除了根據(jù)變量長度來選擇變量類型外,你還要考慮是否變量是否會(huì)用于負(fù)數(shù)的場合。如果你的程序中可以不需要負(fù)數(shù)那么把變量都定義成無符號(hào)類型的。
3 避免使用浮點(diǎn)指針
在 8 位操作系統(tǒng)上使用 32 位浮點(diǎn)數(shù)是得不償失的。你可以這樣做,但會(huì)浪費(fèi)大量的時(shí)間,所以當(dāng)你要在系統(tǒng)中使用浮點(diǎn)數(shù)的時(shí)候,你要問問自己這是否一定需要,可以通過提高數(shù)值數(shù)量級(jí)和使用整型運(yùn)算來消除浮點(diǎn)指針,處理ints和longs比處理doubles和floats要方便得多,你的代碼執(zhí)行起來會(huì)更快,也不用連接處理浮點(diǎn)指針的模塊。如果你一定要,采用浮點(diǎn)指針的話,你應(yīng)該采用西門子 80517 和達(dá)拉斯半導(dǎo)體公司的 80320 這些已經(jīng)對(duì)數(shù),處理進(jìn)行過優(yōu)化的單片機(jī)。如果你不得不在你的代碼中加入浮點(diǎn)指針,那么你的代碼長度會(huì)增加程序執(zhí)行速度也會(huì)比較慢。如果浮點(diǎn)指針運(yùn)算能被中斷的話,你必須確保要么中斷中不會(huì)使用浮點(diǎn)指針運(yùn)算,要么在中斷程序前使用 fpsave 指令把中斷指針推入堆棧,在中斷程序執(zhí)行后使用 fprestore 指令把指針恢復(fù),還有一種方法是,當(dāng)你要使用像 sin()這樣的浮點(diǎn)運(yùn)算程序時(shí),禁止使用中斷,在運(yùn)算程序執(zhí)行完之后再使能它。
4 使用位變量
對(duì)于某些標(biāo)志位應(yīng)使用位變量而不是 unsigned char,這將節(jié)省你的內(nèi)存,你不用多浪費(fèi)7位存儲(chǔ)區(qū),而且位變量在RAM中訪問他們只需要一個(gè)處理周期。
5 用局部變量代替全局變量
變量定義成局部變量比全局變量更有效率,編譯器為局部變量在內(nèi)部存儲(chǔ)區(qū)中分配存儲(chǔ)空間,而為全局變量在外部存儲(chǔ)區(qū)中分配存儲(chǔ)空間,這會(huì)降低你的訪問速度,另一個(gè)避免使用全局變量的原因是你必須在你系統(tǒng)的處理過程中調(diào)節(jié)使用全局變量,因?yàn)樵?strong>中斷系統(tǒng)和多任務(wù)系統(tǒng)中,不止一個(gè)過程會(huì)使用全局變量。
6 為變量分配內(nèi)部存儲(chǔ)區(qū)
局部變量和全局變量可被定義在你想要的存儲(chǔ)區(qū)中,根據(jù)先前的討論,當(dāng)你把經(jīng)常使用的變量放在內(nèi)部 RAM 中時(shí),可使你的程序的速度得到提高,除此之外,你還縮短了你的代碼,因?yàn)橥獠看鎯?chǔ)區(qū)尋址的指令相對(duì)要麻煩一些考慮到存儲(chǔ)速度,按下面的順序使用存儲(chǔ)器DATA IDATA PDATA XDATA,當(dāng)然你要記得留出足夠的堆??臻g。
7 使用特定指針
當(dāng)你在程序中使用指針時(shí),你應(yīng)指定指針的類型確定它們指向哪個(gè)區(qū)域如 XDATA 或CODE 區(qū),這樣你的代碼會(huì)更加緊湊,因?yàn)榫幾g器不必去確定指針?biāo)赶虻拇鎯?chǔ)區(qū),因?yàn)槟阋呀?jīng)進(jìn)行了說明。
8 使用調(diào)令
對(duì)于一些簡單的操作,如變量循環(huán)位移,編譯器提供了一些調(diào)令供用戶使用,許多調(diào)令直接對(duì)應(yīng)著匯編指令,而另外一些比較復(fù)雜并兼容 ANSI 所有這些調(diào)令都是再入函數(shù),你可在任何地方安全的調(diào)用他們和單字節(jié)循環(huán)位移指令 RL A 和 RR A 相對(duì)應(yīng)的調(diào)令是_crol_ 循環(huán)左移 和_cror_(循環(huán)右移)。如果你想對(duì) int 或 long 類型的變量進(jìn)行循環(huán)位移,調(diào)令將更加復(fù)雜而且執(zhí)行的時(shí)間會(huì)更長 對(duì)于 int 類型調(diào)令為_irol_,_iror_ ,對(duì)于 long 類型調(diào)令為_lrol_,_lror_。在 C 中也提供了像匯編中 JBC 指令那樣的調(diào)令_testbit_ ,如果參數(shù)位置位他將返回1,否則將返回 0 這條調(diào)令在檢查標(biāo)志位時(shí)十分有用,而且使 C 的代碼更具有可讀性調(diào)令將直接轉(zhuǎn)換成 JBC 指令。
#include
void serial_intr(void) interrupt 4 {
if (!_testbit_(TI)) { // 是否是發(fā)送中斷
P0=1; // 翻轉(zhuǎn) P0.0
_nop_(); // 等待一個(gè)指令周期
P0=0;
...
}
if (!_testbit_(RI)) {
test=_cror_(SBUF, 1); // 將SBUF中的數(shù)據(jù)循環(huán)
// 右移一位
...
}
}
8 使用宏替代函數(shù)
對(duì)于小段代碼,像使能某些電路或從鎖存器中讀取數(shù)據(jù),你可通過使用宏來替代函數(shù)使得程序有更好的可讀性你可把代碼定義在宏中,這樣看上去更像函數(shù)。編譯器在碰到宏時(shí),按照事先定義的代碼去替代宏,宏的名字應(yīng)能夠描述宏的操作,當(dāng)需要改變宏時(shí),你只要修該宏定義處。
#define led_on() {
led_state=LED_ON;
XBYTE[LED_CNTRL] = 0x01;}
#define led_off() {
led_state=LED_OFF;
XBYTE[LED_CNTRL] = 0x00;}
#define checkvalue(val)
( (val < MINVAL || val > MAXVAL) ? 0 : 1 )
宏能夠使得訪問多層結(jié)構(gòu)和數(shù)組更加容易,可以用宏來替代程序中經(jīng)常使用的復(fù)雜語句以減少你打字的工作量且有更好的可讀性和可維護(hù)性。
9 存儲(chǔ)器模式
C51提供了 3 種存儲(chǔ)器模式來存儲(chǔ)變量、過程參數(shù)和分配再入函數(shù)堆棧。你應(yīng)該盡量使用小存儲(chǔ)器模式,很少應(yīng)用系統(tǒng)需要使用其它兩種模式,像有大的再入函數(shù)堆棧系統(tǒng)那樣。一般來說如果系統(tǒng)所需要的內(nèi)存數(shù)小于內(nèi)部RAM 數(shù)時(shí),都應(yīng)以小存儲(chǔ)模式進(jìn)行編譯。在這種模式下 DATA 段是所有內(nèi)部變量和全局變量的默認(rèn)存儲(chǔ)段,所有參數(shù)傳遞都發(fā)生在DATA 段中,如果有函數(shù)被聲明為再入函數(shù),編譯器會(huì)在內(nèi)部 RAM 中為他們分配空間,這種模式的優(yōu)勢(shì)就是數(shù)據(jù)的存取速度很快,但只有120個(gè)字節(jié)的存儲(chǔ)空間供你使用,總共有128個(gè)字節(jié),但至少有8個(gè)字節(jié)被寄存器組使用,你還要為程序調(diào)用開辟足夠的堆棧。如果你的系統(tǒng)有 256 字節(jié)或更少的外部 RAM 你可以使用壓縮存儲(chǔ)模式。這樣一來,如果不加說明,變量將被分配在 PDATA 段中,這種模式將擴(kuò)充你能夠使用的 RAM 數(shù)量,對(duì)XDATA 段以外的數(shù)據(jù)存儲(chǔ)仍然是很快的,變量的參數(shù)傳遞將在內(nèi)部 RAM 中進(jìn)行,這樣存儲(chǔ)速度會(huì)比較快,對(duì) PDATA 段的數(shù)據(jù)的尋址是通過 R0 和R1進(jìn)行間接尋址,比使用 DPTR 要快一些在大存儲(chǔ)模式中,所有變量的默認(rèn)存儲(chǔ)區(qū)是 XDATA 段 Keil C 盡量使用內(nèi)部寄存器組進(jìn)行參數(shù)傳遞,在寄存器組中可以傳遞參數(shù)的數(shù)量和和壓縮存儲(chǔ)模式一樣,再入函數(shù)的模擬棧將在 XDATA中 對(duì) XDATA 段數(shù)據(jù)的訪問是最慢的,所以要仔細(xì)考慮變量應(yīng)存儲(chǔ)的位置使數(shù)據(jù)的存儲(chǔ)速度得到優(yōu)化。
10 混合存儲(chǔ)模式
Keil 允許使用混合的存儲(chǔ)模式,這點(diǎn)在大存儲(chǔ)模式中是非常有用的。在大存儲(chǔ)器模式下,有些過程對(duì)數(shù)據(jù)傳遞的速度要求很高。我就把過程定義在小存儲(chǔ)模式寄存器中,這使得編譯器為該過程的局部變量在內(nèi)部 RAM中分配存儲(chǔ)空間,并保證所有參數(shù)都通過內(nèi)部 RAM進(jìn)行傳遞。盡管采用混合模式后編譯的代碼長度不會(huì)有很大的改變,但這種努力是值得的就像能在大模式下把過程聲明為小模式一樣,你像能在小模式下把過程聲明為壓縮?;虼竽J?,這一般使用在需要大量存儲(chǔ)空間的過程上,這樣過程中的局部變量將被存儲(chǔ)在外部存儲(chǔ)區(qū)中,你也可以通過過程中的變量聲明,把變量分配在 XDATA 段中。
11 運(yùn)行庫
運(yùn)行庫中提供了很多短小精悍的函數(shù),你可以很方便的使用他們,你自己很難寫出更好的代碼了。值得注意的是庫中有些函數(shù)不是再入函數(shù),如果在執(zhí)行這些函數(shù)的時(shí)候被中斷,而在中斷程序中又調(diào)用了該函數(shù),將得到意想不到的結(jié)果。而且這種錯(cuò)誤很難找出來,最好禁止使用這些函數(shù)的中斷。



評(píng)論


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

關(guān)閉