找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開(kāi)始

搜索
查看: 3913|回復(fù): 0
收起左側(cè)

51單片機(jī)的data,bdata,idata,pdata,xdata,code存儲(chǔ)類(lèi)型與存儲(chǔ)區(qū)bit

[復(fù)制鏈接]
ID:105323 發(fā)表于 2016-2-13 02:27 | 顯示全部樓層 |閱讀模式
  bit是在內(nèi)部數(shù)據(jù)存儲(chǔ)空間中 20H .. 2FH區(qū)域中一個(gè)位的地址,這在DATA的20H以后以字節(jié)形式出現(xiàn),可互相參照。另外加上8051 可尋址的SFR,但剛剛試過(guò),只是00H--7FH起作用,也就是說(shuō)當(dāng)數(shù)據(jù)有變化時(shí)顏色變紅,以后的從80H到--FFH就不是位尋址區(qū)了,是位尋址的特殊寄存器,如涉及到了可位尋址的那11個(gè)當(dāng)然會(huì)有反應(yīng)。
復(fù)位后,程序計(jì)數(shù)器PC的內(nèi)容為0000H,內(nèi)部RAM各單元的值不確定。各功能寄存器的復(fù)位值如下:堆棧指針SP的復(fù)位值為07H,累加器ACC、寄存器B的復(fù)位值為00H,數(shù)據(jù)指針DPTR的復(fù)位值為0000H,而p0、p1、p2、p3四個(gè)口的復(fù)位值為0FFH。其他SFR如PSW、TCON、TMOD、TL0、TH0、TL1、TH1的復(fù)位值也為00H。
wave中是低128字節(jié)和高128字節(jié)(0-7FH),低128字節(jié)是片內(nèi)RAM區(qū),高128字節(jié)(80-FFH)是SFR(特殊功能寄存器)bit則是位于低128字節(jié)的20H.. 2FH 區(qū)域,即data的20H .. 2FH 區(qū)域

code
是在 0000H .. 0FFFFH 之間的一個(gè)代碼地址。
我用
ORG    5000H
TAB:   DB     22H,3BH,43H,66H,5H,6DH,88H后,
CODE從5000H開(kāi)始以后變成DB各位

data

是在 0 到 127 之間的一個(gè)數(shù)據(jù)存儲(chǔ)器地址,或者加 128 .. 255范圍內(nèi)的一個(gè)特殊功能寄存器(SFR)地址。兩者訪問(wèn)的方式不同。實(shí)際上由于PSW的復(fù)位設(shè)置PSW.3=RS0和PSW.4=RS1皆為0,所以通用工作寄存器區(qū)就是第0區(qū),所以data的00--07H部分是與REG欄中的R0--R7對(duì)應(yīng)的。以后的則僅代表低128字節(jié)的內(nèi)部RAM。


idata
是 0 to 255 范圍內(nèi)的一個(gè) idata 存儲(chǔ)器地址。
idata與data重合低128字節(jié),有的地方只有DATA表示256字節(jié)的片內(nèi)RAM,
xdata 是 0 to 65535 范圍內(nèi)的一個(gè) xdata 存儲(chǔ)器地址。


指針類(lèi)型和存儲(chǔ)區(qū)的關(guān)系詳解

一、存儲(chǔ)類(lèi)型與存儲(chǔ)區(qū)關(guān)系

    data    --->    可尋址片內(nèi)ram
    bdata    --->    可位尋址的片內(nèi)ram
    idata    --->    可尋址片內(nèi)ram,允許訪問(wèn)全部?jī)?nèi)部ram
    pdata    --->    分頁(yè)尋址片外ram (MOVX @R0) (256 BYTE/頁(yè))
    xdata    --->    可尋址片外ram(64k地址范圍FFFFH)
    code    --->    程序存儲(chǔ)區(qū)(64k 地址范圍),對(duì)應(yīng)MOVC @DPTR

二、指針類(lèi)型和存儲(chǔ)區(qū)的關(guān)系

    對(duì)變量進(jìn)行聲明時(shí)可以指定變量的存儲(chǔ)類(lèi)型如:
    uchar data x和data uchar x相等價(jià)都是在內(nèi)ram區(qū)分配一個(gè)字節(jié)的變量。

    同樣對(duì)于指針變量的聲明,因涉及到指針變量本身的存儲(chǔ)位置和指針?biāo)赶虻拇鎯?chǔ)區(qū)位置不同而進(jìn)行相應(yīng)的存儲(chǔ)區(qū)類(lèi)型關(guān)鍵字的
使用如:

    uchar xdata * data pstr

    是指在內(nèi)ram區(qū)分配一個(gè)指針變量("*"號(hào)后的data關(guān)鍵字的作用),而且這個(gè)指針本身指向xdata區(qū)("*"前xdata關(guān)鍵字的作用),
可能初學(xué)C51時(shí)有點(diǎn)不好懂也不好記。沒(méi)關(guān)系,我們馬上就可以看到對(duì)應(yīng)“*”前后不同的關(guān)鍵字的使用在編譯時(shí)出現(xiàn)什么情況。

    ......
    uchar xdatatmp[10];    //在外ram區(qū)開(kāi)辟10個(gè)字節(jié)的內(nèi)存空間,地址是外ram的0x0000-0x0009
    ......

    第1種情況:

    uchar data * data pstr;
    pstr=tmp;

    首先要提醒大家這樣的代碼是有bug的, 他不能通過(guò)這種方式正確的訪問(wèn)到tmp空間。 為什么?我們把編譯后看到下面的匯編
代碼:

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

    看到了嗎!本來(lái)訪問(wèn)外ram需要2byte來(lái)尋址64k空間,但因?yàn)槭褂胐ata關(guān)鍵字(在"*"號(hào)前的那個(gè)),所以按KeilC編譯環(huán)境來(lái)說(shuō)
就把他編譯成指向內(nèi)ram的指針變量了,這也是初學(xué)C51的朋友們不理解各個(gè)存儲(chǔ)類(lèi)型的關(guān)鍵字定義而造成的bug。特別是當(dāng)工程中的
默認(rèn)的存儲(chǔ)區(qū)類(lèi)為large時(shí),又把tmp[10] 聲明為uchar tmp[10]時(shí),這樣的bug是很隱秘的不容易被發(fā)現(xiàn)。

    第2種情況:

    uchar xdata * data pstr;
    pstr = tmp;

    這種情況是沒(méi)問(wèn)題的,這樣的使用方法是指在內(nèi)ram分配一個(gè)指針變量("*"號(hào)后的data關(guān)鍵字的作用),而且這個(gè)指針本身指向
xdata區(qū)("*"前xdata關(guān)鍵字的作用)。編譯后的匯編代碼如下。

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

    這種情況應(yīng)該是在這里所有介紹各種情況中效率最高的訪問(wèn)外ram的方法了,請(qǐng)大家記住他。

    第3種情況:

    uchar xdata * xdata pstr;
    pstr=tmp;

    這中情況也是對(duì)的,但效率不如第2種情況。編譯后的匯編代碼如下。

    MOV DPTR,#0x000A        ;0x000A,0x000B是在外ram區(qū)分配的pstr指針變量地址空間
    MOV A, #tmp(0x00)
    MOV @DPTR, A
    INC DPTR
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    這種方式一般用在內(nèi)ram資源相對(duì)緊張而且對(duì)效率要求不高的項(xiàng)目中。

    第4種情況:

    uchar data * xdata pstr;
    pstr=tmp;

    如果詳細(xì)看了第1種情況的讀者發(fā)現(xiàn)這種寫(xiě)法和第1種很相似,是的,同第1 種情況一樣這樣也是有bug的,但是這次是把pstr分
配到了外ram區(qū)了。編譯后的匯編代碼如下。

    MOV DPTR,#0x000A        ;0x000A是在外ram區(qū)分配的pstr指針變量的地址空間
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    第5種情況:

    uchar * data pstr;
    pstr=tmp;

    大家注意到"*"前的關(guān)鍵字聲明沒(méi)有了,是的這樣會(huì)發(fā)生什么事呢?下面這么寫(xiě)呢!對(duì)了用齊豫的一首老歌名來(lái)說(shuō)就是 “請(qǐng)跟我
來(lái)”,請(qǐng)跟我來(lái)看看編譯后的匯編代碼,有人問(wèn)這不是在講C51嗎?為什么還要給我們看匯編代碼。C51要想用好就要盡可能提升C51
編譯后的效率,看看編譯后的匯編會(huì)幫助大家盡快成為生產(chǎn)高效C51代碼的高手的。還是看代碼吧!

    MOV 0x08,#0X01            ;0x08-0x0A是在內(nèi)ram區(qū)分配的pstr指針變量的地址空間
    MOV 0x09, #tmp(0x00)
    MOV 0x0A, #tmp(0x00)

    注意:這是新介紹給大家的,大家會(huì)疑問(wèn)為什么在前面的幾種情況的pstr指針變量都用2 byte空間而到這里就用3byte空間了
呢?這是KeilC的一個(gè)系統(tǒng)內(nèi)部處理,在KeilC中一個(gè)指針變量最多占用 3byte空間,對(duì)于沒(méi)有聲明指針指向存儲(chǔ)空間類(lèi)型的指針,
系統(tǒng)編譯代碼時(shí)都強(qiáng)制加載一個(gè)字節(jié)的指針類(lèi)型分辯值。具體的對(duì)應(yīng)關(guān)系可以參考KeilC的help中C51 User'sGuide。

    第6種情況:

    uchar * pstr;
    pstr=tmp;

    這是最直接最簡(jiǎn)單的指針變量聲明,但他的效率也最低。還是那句話,大家一起說(shuō)好嗎!編譯后的匯編代碼如下。

    MOV DPTR,#0x000A        ;0x000A-0x000C是在外ram區(qū)分配的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

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



相關(guān)帖子

回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表