C51存儲器類型與51單片機(jī)的物理區(qū)域
1、 data區(qū)空間小,所以只有頻繁用到或?qū)\算速度要求很高的變量才放到data區(qū)內(nèi),比如for循環(huán)中的計數(shù)值。
2、 data區(qū)內(nèi)最好放局部變量。
因為局部變量的空間是可以覆蓋的(某個函數(shù)的局部變量空間在退出該函數(shù)是就釋放,由別的函數(shù)的局部變量覆蓋),可以提高內(nèi)存利用率。當(dāng)然靜態(tài)局部變量除外,其內(nèi)存使用方式與全局變量相同;
3、 確保你的程序中沒有未調(diào)用的函數(shù)。
在Keil C里遇到未調(diào)用函數(shù),編譯器就將其認(rèn)為可能是中斷函數(shù)。函數(shù)里用的局部變量的空間是不釋放,也就是同全局變量一樣處理。這一點Keil C做得很愚蠢,但也沒辦法。
4、 程序中遇到的邏輯標(biāo)志變量可以定義到bdata中,可以大大降低內(nèi)存占用空間。
在51系列芯片中有16個字節(jié)位尋址區(qū)bdata,其中可以定義8*16=128個邏輯變量。定義方法是: bdata bit LedState;但位類型不能用在數(shù)組和結(jié)構(gòu)體中。
5、 其他不頻繁用到和對運算速度要求不高的變量都放到xdata區(qū)。
6、 如果想節(jié)省data空間就必須用large模式,將未定義內(nèi)存位置的變量全放到xdata區(qū)。當(dāng)然最好對所有變量都要指定內(nèi)存類型。
7、 當(dāng)使用到指針時,要指定指針指向的內(nèi)存類型。在C51中未定義指向內(nèi)存類型的通用指針占用3個字節(jié);而指定指向data區(qū)的指針只占1個字節(jié);指定指向xdata區(qū)的指針占2個字節(jié)。如指針p是指向data區(qū),則應(yīng)定義為: char data *p;。還可指定指針本身的存放內(nèi)存類型,如:char data * xdata p;。其含義是指針p指向data區(qū)變量,而其本身存放在xdata區(qū)。
bit---是在內(nèi)部數(shù)據(jù)存儲空間中 20H .. 2FH 區(qū)域中一個位的地址,或者 8051 位可尋址 SFR 的一個位地址。
code---是在 0000H .. 0FFFFH 之間的一個代碼地址。
data---是在 0 到 127 之間的一個數(shù)據(jù)存儲器地址,或者在 128 .. 255 范圍內(nèi)的一個特殊功能寄存器(SFR)地址。
idata---是 0 to 255 范圍內(nèi)的一個 idata 存儲器地址。
xdata 是 0 to 65535 范圍內(nèi)的一個 xdata 存儲器地址。
指針類型和存儲區(qū)的關(guān)系詳解
一、存儲類型與存儲區(qū)關(guān)系
data ---> 可尋址片內(nèi)ram
bdata ---> 可位尋址的片內(nèi)ram
idata ---> 可尋址片內(nèi)ram,允許訪問全部內(nèi)部ram
pdata ---> 分頁尋址片外ram (MOVX @R0) (256 BYTE/頁)
xdata ---> 可尋址片外ram (64k 地址范圍)
code ---> 程序存儲區(qū) (64k 地址范圍),對應(yīng)MOVC @DPTR
二、指針類型和存儲區(qū)的關(guān)系
對變量進(jìn)行聲明時可以指定變量的存儲類型如:
uchar data x和data uchar x相等價都是在內(nèi)ram區(qū)分配一個字節(jié)的變量。
同樣對于指針變量的聲明,因涉及到指針變量本身的存儲位置和指針?biāo)赶虻拇鎯^(qū)位置不同而進(jìn)行相應(yīng)的存儲區(qū)類型關(guān)鍵字的使用如:
uchar xdata * data pstr
是指在內(nèi)ram區(qū)分配一個指針變量("*"號后的data關(guān)鍵字的作用),而且這個指針本身指向xdata區(qū)("*"前xdata關(guān)鍵字的作用),可能初學(xué)C51時有點不好懂也不好記。沒關(guān)系,我們馬上就可以看到對應(yīng)“*”前后不同的關(guān)鍵字的使用在編譯時出現(xiàn)什么情況。
......
uchar xdata tmp[10]; //在外ram區(qū)開辟10個字節(jié)的內(nèi)存空間,地址是外ram的0x0000-0x0009
......
第1種情況:
uchar data * data pstr;
pstr=tmp;
首先要提醒大家這樣的代碼是有bug的, 他不能通過這種方式正確的訪問到tmp空間。 為什么?我們把編譯后看到下面的匯編代碼:
MOV 0x08,#tmp(0x00) ;0x08是指針pstr的存儲地址
看到了嗎!本來訪問外ram需要2 byte來尋址64k空間,但因為使用data關(guān)鍵字(在"*"號前的那個),所以按KeilC編譯環(huán)境來說就把他編譯成指向內(nèi)ram的指針變量了,這也是初學(xué)C51的朋友們不理解各個存儲類型的關(guān)鍵字定義而造成的bug。特別是當(dāng)工程中的默認(rèn)的存儲區(qū)類為large時,又把tmp[10] 聲明為uchar tmp[10] 時,這樣的bug是很隱秘的不容易被發(fā)現(xiàn)。
評論