C51存儲器類型與51單片機的物理區域

1、 data區空間小,所以只有頻繁用到或對運算速度要求很高的變量才放到data區內,比如for循環中的計數值。

2、 data區內最好放局部變量。

因爲局部變量的空間是可以覆蓋的(某個函數的局部變量空間在退出該函數是就釋放,由別的函數的局部變量覆蓋),可以提高內存利用率。當然靜態局部變量除外,其內存使用方式與全局變量相同;

擴展閱讀:51單片機的特殊功能寄存器(SFR

3、 確保你的程序中沒有未調用的函數。

在Keil C裏遇到未調用函數,編譯器就將其認爲可能是中斷函數。函數裏用的局部變量的空間是不釋放,也就是同全局變量一樣處理。這一點Keil C做得很愚蠢,但也沒辦法。

4、 程序中遇到的邏輯標誌變量可以定義到bdata中,可以大大降低內存佔用空間。

在51系列芯片中有16個字節位尋址區bdata,其中可以定義8*16=128個邏輯變量。定義方法是: bdata bit LedState;但位類型不能用在數組和結構體中。

5、 其他不頻繁用到和對運算速度要求不高的變量都放到xdata區。

6、 如果想節省data空間就必須用large模式,將未定義內存位置的變量全放到xdata區。當然最好對所有變量都要指定內存類型。

7、 當使用到指針時,要指定指針指向的內存類型。在C51中未定義指向內存類型的通用指針佔用3個字節;而指定指向data區的指針只佔1個字節;指定指向xdata區的指針佔2個字節。如指針p是指向data區,則應定義爲: char data *p;。還可指定指針本身的存放內存類型,如:char data * xdata p;。其含義是指針p指向data區變量,而其本身存放在xdata區。

bit---是在內部數據存儲空間中 20H .. 2FH 區域中一個位的地址,或者 8051 位可尋址 SFR 的一個位地址。

code---是在 0000H .. 0FFFFH 之間的一個代碼地址。

data---是在 0 到 127 之間的一個數據存儲器地址,或者在 128 .. 255 範圍內的一個特殊功能寄存器(SFR)地址。

idata---是 0 to 255 範圍內的一個 idata 存儲器地址。

xdata 是 0 to 65535 範圍內的一個 xdata 存儲器地址。

指針類型和存儲區的關係詳解

一、存儲類型與存儲區關係

data ---> 可尋址片內ram

bdata ---> 可位尋址的片內ram

idata ---> 可尋址片內ram,允許訪問全部內部ram

pdata ---> 分頁尋址片外ram (MOVX @R0) (256BYTE/頁)

xdata ---> 可尋址片外ram (64k 地址範圍)

code ---> 程序存儲區 (64k 地址範圍),對應MOVC @DPTR

二、指針類型和存儲區的關係

對變量進行聲明時可以指定變量的存儲類型如:

uchar data x和data uchar x相等價都是在內ram區分配一個字節的變量。

同樣對於指針變量的聲明,因涉及到指針變量本身的存儲位置和指針所指向的存儲區位置不同而進行相應的存儲區類型關鍵字的使用如:

uchar xdata * data pstr

是指在內ram區分配一個指針變量("*"號後的data關鍵字的作用),而且這個指針本身指向xdata區("*"前xdata關鍵字的作用),可能初學C51時有點不好懂也不好記。沒關係,我們馬上就可以看到對應“*”前後不同的關鍵字的使用在編譯時出現什麼情況。

......

uchar xdata tmp[10]; //在外ram區開闢10個字節的內存空間,地址是外ram的0x0000-0x0009

......

第1種情況:

uchar data * data pstr;

pstr=tmp;

首先要提醒大家這樣的代碼是有bug的, 他不能通過這種方式正確的訪問到tmp空間。 爲什麼?我們把編譯後看到下面的彙編代碼:

MOV 0x08,#tmp(0x00) ;0x08是指針pstr的存儲地址

看到了嗎!本來訪問外ram需要2 byte來尋址64k空間,但因爲使用data關鍵字(在"*"號前的那個),所以按KeilC編譯環境來說就把他編譯成指向內ram的指針變量了,這也是初學C51的朋友們不理解各個存儲類型的關鍵字定義而造成的bug。特別是當工程中的默認的存儲區類爲large時,又把tmp[10] 聲明爲uchar tmp[10] 時,這樣的bug是很隱祕的不容易被發現。

第2種情況:

uchar xdata * data pstr;

pstr = tmp;

這種情況是沒問題的,這樣的使用方法是指在內ram分配一個指針變量("*"號後的data關鍵字的作用),而且這個指針本身指向xdata區("*"前xdata關鍵字的作用)。編譯後的彙編代碼如下。

MOV 0x08,#tmp(0x00) ;0x08和0x09是在內ram區分配的pstr指針變量地址空間

MOV 0x09,#tmp(0x00)

這種情況應該是在這裏所有介紹各種情況中效率最高的訪問外ram的方法了,請大家記住他。

第3種情況:

uchar xdata * xdata pstr;

pstr=tmp;

這中情況也是對的,但效率不如第2種情況。編譯後的彙編代碼如下。

MOV DPTR, #0x000A ;0x000A,0x000B是在外ram區分配的pstr指針變量地址空間

MOV A, #tmp(0x00)

MOV @DPTR, A

INC DPTR

MOV A, #tmp(0x00)

MOVX @DPTR, A

這種方式一般用在內ram資源相對緊張而且對效率要求不高的項目中。

第4種情況:

uchar data * xdata pstr;

pstr=tmp;

如果詳細看了第1種情況的讀者發現這種寫法和第1種很相似,是的,同第1 種情況一樣這樣也是有bug的,但是這次是把pstr分

配到了外ram區了。編譯後的彙編代碼如下。

MOV DPTR, #0x000A ;0x000A是在外ram區分配的pstr指針變量的地址空間

MOV A, #tmp(0x00)

MOVX @DPTR, A

第5種情況:

uchar * data pstr;

pstr=tmp;

大家注意到"*"前的關鍵字聲明沒有了,是的這樣會發生什麼事呢?下面這麼寫呢!對了用齊豫的一首老歌名來說就是 “請跟我來”,請跟我來看看編譯後的彙編代碼,有人問這不是在講C51嗎? 爲什麼還要給我們看彙編代碼。C51要想用好就要儘可能提升C51編譯後的效率,看看編譯後的彙編會幫助大家儘快成爲生產高效C51代碼的高手的。還是看代碼吧!

MOV 0x08, #0X01 ;0x08-0x0A是在內ram區分配的pstr指針變量的地址空間

MOV 0x09, #tmp(0x00)

MOV 0x0A, #tmp(0x00)

注意:這是新介紹給大家的,大家會疑問爲什麼在前面的幾種情況的pstr指針變量都用2 byte空間而到這裏就用3 byte空間了呢?這是KeilC的一個系統內部處理,在KeilC中一個指針變量最多佔用 3 byte空間,對於沒有聲明指針指向存儲空間類型的指針,系統編譯代碼時都強制加載一個字節的指針類型分辯值。具體的對應關係可以參考KeilC的help中C51User's Guide。

第6種情況:

uchar * pstr;

pstr=tmp;

這是最直接最簡單的指針變量聲明,但他的效率也最低。還是那句話,大家一起說好嗎!編譯後的彙編代碼如下。

MOV DPTR, #0x000A ;0x000A-0x000C是在外ram區分配的pstr指針變量地址空間

MOV A, #0x01

MOV @DPTR, A

INC DPTR

MOV DPTR, #0x000A

MOV A, #tmp(0x00)

MOV @DPTR, A

INC DPTR

MOV A, #tmp(0x00)

MOVX @DPTR, A

這種情況很類似第5種和第3種情況的組合,既把pstr分配在外ram空間了又增加了指針類型的分辨值。

小結一下:大家看到了以上的6種情況,其中效率最高的是第2種情況,既可以正確訪問ram區又節約了代碼,效率最差的是第 6種,但不是說大家只使用第2種方式就可以了,還要因情況而定,一般說來應用51系列的系統架構的內部ram資源都很緊張,最好大家在定義函數內部或程序段內部的局部變量使用內ram,而儘量不要把全局變量聲明爲內ram區中。所以對於全局指針變量我建議使用第3 種情況,而對於局部的指針變量使用第2種方式。

keil C51存儲區域分爲程序存儲區和數據存儲區2大類型。

一.程序存儲區(Pragram Area):

欲將聲明的數據存放在程序存儲區域,可以使用關鍵字“code”說明。

例 unsigned char code i=10;則表示 i爲無符號字符型數據存放區域爲程序存儲區。

二.數據存儲區(Data Memory):

數據存儲區域分爲內部數據存儲區、外部數據存儲區域和特殊功能寄存器尋址區。

1.內部數據存儲區域(Internal Data Memory):可以使關鍵字"data、iadta、bdata"做相應說明。

data:直接尋址區,聲明的數據存儲範圍爲內部RAM低128字節 0X00~0X7F。

例 unsigned char data i=10;則表示 i爲無符號字符型數據存放區域爲數據存儲區域(RAM)的低128字節範圍內。

idata:間接尋址區,聲明的數據存儲範圍爲整個內部RAM區 0X00~0XFF。

例 unsigned char idatai=10;則表示 i爲無符號字符型數據存放區域爲數據存儲區域(RAM)內。

bdata:可位尋址區,尋址範圍爲0X20~0X2F。

2.外部數據存儲區(External Data Memory):可以使用關鍵字"pdata、xdata"進行說明。

pdata:主要用於緊湊模式,能訪問1頁(256字節)的外部RAM,即在訪問使用,pdata定義的數據時,不會影響P2口的輸出電平(在訪問某些自身內部擴展的外部RAM時本身就不會影響I/O端口)。

例 unsigned char pdata i;則表示 i爲無符號字符型數據存放區域爲外部數據存儲區域(RAM)內(只能在一頁範圍內)具體操作哪一頁,可由其他i/o口設定。

xdata:可訪問64k的外部數據存儲區,地址範圍0X0000~0XFFFF,同pdata一樣在訪問芯片自身內部擴展的RAM時也不會影響I/O端口。

例 unsigned char pdata i;則表示 i爲無符號字符型數據存放區域爲外部數據存儲區(RAM)。

3.特殊功能寄存器尋址區域(Speciac Function Register Memory)-SFR:8051提供128字節的SFR尋址區,該區域可以字節尋址,字尋址,能被8整除的地址單元還可以位尋址。該區域用於控制定時器、計數器、串口等外圍接口。使用時可用關鍵字“sfr、sfr16、sbit”做相應的聲明。

例 字節尋址 sfr P0=0x80;爲P0 口地址爲80H,“=”後0X00~0XFF 之間的常數。

字尋址 sfr16 T2=0XCC;指定Timer2 口地址T2L=0XCC T2H=0XCD。

位尋址 sbit EA=0xAF;指定第0xAF 位爲EA,即中斷允許

存儲模式

在使用C51時有時我們並沒有明確指定所定義的數據的存儲類型,然而依然正確。這是由於存儲模式決定了沒有明確指定存儲類型的變量,函數參數等的缺省存儲區域。

供有3種存儲模式(存儲模式在 C51 編譯器選項中選擇):

1.Small模式

所有缺省變量參數均裝入內部 RAM,優點是訪問速度快,缺點是空間有限,只適用於小程序。

2. Compact 模式

所有缺省變量均位於外部RAM區的一頁(256Byte)。

3. Large 模式

所有缺省變量可放在多達 64KB 的外部RAM 區,優點是空間大,可存變量多,缺點是速度較慢。

C51存儲器類型有bit sbit data xdata bdata pdata sfr code等,可能不全面有遺漏

對應的物理存儲器是:

bit,即位數據:數據存儲器位尋址區,即20H~2FH的範圍,共16個字節,16*8=128個位,位地址00h~7fh,連續的。

sbit:特殊功能寄存器中的位數據:只有能夠被8整除的那些特殊功能寄存器中的各個位才能被稱爲sbit,位地址80H~FFH,不連續的,間斷的。

data:數據區,對51爲00H~7FH共128個字節,對52爲00H~FFH,共256個字節,用MOV尋址,前128用直接尋址或寄存器(R0~R7)尋址,後128用R0、R1間接尋址。

xdata:外部數據區,0000H~FFFFH連續,用DPTR間接尋址(MOVX指令)

bdata:位尋址去的字節,20H~2FH

sfr:特殊功能寄存器(80H~FFH),直接尋址

pdata:外部數據區,P2口保持數據,用R0R1間接尋址(MOVX指令)

code:程序存儲器,用MOVC指令只讀

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章