單片機data,bdata,idata,pdata,xdata,code

普通51系列單片機存儲空間資源分配情況

空間名稱
地址範圍
說明
DATA
D:00H~7FH
片內RAM直接尋址區
BDATA
D:20H~2FH
片內RAM位尋址區
IDATA
I:00H~FFH
片內RAM間接尋址區
XDATA
X:0000H~FFFFH
64K片外RAM數據區
CODE
C:0000H~FFFFH
64K片內外ROM代碼區
BANK0~BANK31
B0:0000H~FFFFH
:
:
B31:0000H~FFFFH
分組代碼區,最大可擴展32X64KB ROM

單片機學習網http://www.51hei.com給大家整理出了上下這兩個表格,方便大家查詢。有問題可在評論或者論壇中提出。

下表是新型80C51單片機擴展空間的分配情況

空間名稱
地址範圍
說明
DATA
D:00H~7FH
片內RAM直接尋址區
BDATA
D:20H~2FH
片內RAM位尋址區
IDATA
I:00H~FFH
片內RAM間接尋址區
XDATA
X:0000H~FFFFH
64KB常規片外RAM數據區
HDATA
X:0000H~FFFFFFH
16MB擴展片外RAM數據區
CODE
C:0000H~FFFFH
64K常規片內外ROM代碼區
HCONST(ECODE)
C:0000H~FFFFFFH
16MB擴展片外ROM常數區(對Dallas390可用作代碼區)
BANK0~BANK31
B0:0000H~FFFFH
:
:
B31:0000H~FFFFH
分組代碼區,最大可擴展32X64KB ROM

單片機中關鍵字data,idata,xdata,pdata的區別

data:
固定指前面0x00-0x7f的128個RAM,可以用acc直接讀寫的,速度最快,生成的代碼也最小。

idata:
固定指前面0x00-0xff的256個RAM,其中前128和data的128完全相同,只是因爲訪問的方式不同。idata是用類似C中的指針方式訪問的。彙編中的語句爲:moxACC,@Rx.(不重要的補充:c中idata做指針式的訪問效果很好)

xdata:
外 部擴展RAM,一般指外部0x0000-0xffff空間,用DPTR訪問。

pdata:外部擴展RAM的低256個字節,地址出現在A0-A7的上時讀寫,用movxACC,@Rx讀寫。這個比較特殊,而且C51好象有對此BUG, 建議少用。



如果省略存儲器類型,系統則會按編譯模式SMALL,COMPACT或LARGE所規定的默認存儲器類型去指定變量的存儲區域。無論什麼存儲模式都能聲明變量在任何的8051存儲區範圍,然而把最常用的命令如循環計數器和隊列索引放在內部數據區能顯著的提高系統性能。還有要指出的就是變量的存儲種類與存儲器類型是完全無關的。

. 數據存儲模式
存儲模式決定了沒有明確指定存儲類型的變量,函數參數等的缺省存儲區域,共三種:
1. 1. Small模式
所有缺省變量參數均裝入內部RAM,優點是訪問速度快,缺點是空間有限,只適用於小程序。
2. 2. Compact模式
所有缺省變量均位於外部RAM區的一頁(256Bytes),具體哪一頁可由P2口指定,在STARTUP.A51文件中說明,也可用pdata指定,優點是空間較Small爲寬裕速度較Small慢,較large要快,是一種中間狀態。
3. 3. large模式
所有缺省變量可放在多達64KB的外部RAM區,優點是空間大,可存變量多,缺點是速度較慢。
提示:存儲模式在單片機c語言編譯器選項中選擇。

  之前提到簡單提到sfr,sfr16,sbit定義變量的方法,下面我們再來仔細看看。
  sfr和sfr16能直接對51單片機的特殊寄存器進行定義,定義方法如下:
    sfr 特殊功能寄存器名= 特殊功能寄存器地址常數;
    sfr16 特殊功能寄存器名= 特殊功能寄存器地址常數;
  我們能這樣定義AT89c51的P1口
    sfr P1 = 0x90; //定義P1 I/O口,其地址90H
  sfr關鍵定後面是一個要定義的名字,可任意選取,但要符合標識符的命名規則,名字最好有一定的含義如P1口能用P1爲名,這樣程序會變的好讀好多。等號後面必須是常數,不允許有帶運算符的表達式,而且該常數必須在特殊功能寄存器的地址範圍之內(80H-FFH),具體可查看附錄中的相關表。sfr是定義8位的特殊功能寄存器而sfr16則是用來定義16位特殊功能寄存器,如8052的T2定時器,能定義爲:
    sfr16 T2 = 0xCC; //這裏定義8052定時器2,地址爲T2L=CCH,T2H=CDH
用sfr16定義16位特殊功能寄存器時,等號後面是它的低位地址,高位地址一定要位於物理低位地址之上。注意的是不能用於定時器0和1的定義。
  sbit可定義可位尋址對象。如訪問特殊功能寄存器中的某位。其實這樣應用是經常要用的如要訪問P1口中的第2個引腳P1.1。我們能照以下的方法去定義:
(1)sbit 位變量名=位地址
  sbit P1_1 = Ox91;
這樣是把位的絕對地址賦給位變量。同sfr一樣sbit的位地址必須位於80H-FFH之間。
(2)Sbit 位變量名=特殊功能寄存器名^位位置
sft P1 = 0x90;
  sbit P1_1 = P1 ^ 1; //先定義一個特殊功能寄存器名再指定位變量名所在的位置
當可尋址位位於特殊功能寄存器中時可採用這種方法
(3)sbit 位變量名=字節地址^位位置
  sbit P1_1 = 0x90 ^ 1;
  這種方法其實和2是一樣的,只是把特殊功能寄存器的位址直接用常數表示。
  在單片機c語言存儲器類型中供給有一個bdata的存儲器類型,這個是指可位尋址的數據存儲器,位於單片機的可位尋址區中,能將要求可位錄址的數據定義爲bdata,如:
unsigned char bdata ib; //在可位錄址區定義ucsigned char類型的變量ib
int bdata ab[2]; //在可位尋址區定義數組ab[2],這些也稱爲可尋址位對象
sbit ib7=ib^7 //用關鍵字sbit定義位變量來獨立訪問可尋址位對象的其中一位
sbit ab12=ab[1]^12;
  操作符"^"後面的位位置的最大值取決於指定的基址類型,char0-7,int0-15,long0-31。
下面我們用上一課的電路來實踐一下這一課的知識。同樣是做一下簡單的跑馬燈實驗,項目名爲RunLED2。程序如下:

sfr P1 = 0x90; //這裏沒有使用預定義文件,
sbit P1_0 = P1 ^ 0; //而是自己定義特殊寄存器
sbit P1_7 = 0x90 ^ 7; //之前我們使用的預定義文件其實就是這個作用
sbit P1_1 = 0x91; //這裏分別定義P1端口和P10,P11,P17引腳

void main(void)
{
unsigned int a;
unsigned char b;
do{
for (a=0;a<50000;a++)
P1_0 = 0; //點亮P1_0
for (a=0;a<50000;a++)
P1_7 = 0; //點亮P1_7
for (b=0;b<255;b++)
{
for (a=0;a<10000;a++)
P1 = b; //用b的值來做跑馬燈的花樣
}
P1 = 255; //熄滅P1上的LED
for (b=0;b<255;b++)
{
for (a=0;a<10000;a++) //P1_1閃爍
P1_1 = 0;
for (a=0;a<10000;a++)
P1_1 = 1;
}
}while(1);
}


. Keil c51指針變量
單片機c語言支持一般指針(Generic Pointer)和存儲器指針(Memory_Specific Pointer).
1. 1. 一般指針
一般指針的聲明和使用均與標準C相同,不過同時還能說明指針的存儲類型,例如:
long * state;爲一個指向long型整數的指針,而state本身則依存儲模式存放。
char * xdata ptr;ptr爲一個指向char數據的指針,而ptr本身放於外部RAM區,以上的long,char等指針指向的數據可存放於任何存儲器中。
一般指針本身用3個字節存放,分別爲存儲器類型,高位偏移,低位偏移量。
2. 2. 存儲器指針
基於存儲器的指針說明時即指定了存貯類型,例如:
char data * str;str指向data區中char型數據
int xdata * pow; pow指向外部RAM的int型整數。
這種指針存放時,只需一個字節或2個字節就夠了,因爲只需存放偏移量。
3. 3. 指針轉換
即指針在上兩種類型之間轉化:
l 當基於存儲器的指針作爲一個實參傳遞給需要一般指針的函數時,指針自動轉化。
l 如果不說明外部函數原形,基於存儲器的指針自動轉化爲一般指針,導致錯誤,因而請用“#include”說明所有函數原形。
l 能強行改變指針類型。

變量的存儲類別

一、static(靜態局部)變量。
1、靜態局部變量在程序整個運行期間都不會釋放內存。
2、對於靜態局部變量,是在編譯的時候賦初值的,即只賦值一次。如果在程序運行時已經有初值,則以後每次調用的時候不再重新賦值。
3、如果定義局部變量的時候不賦值,則編譯的時候自動賦值爲0。而對於自動變量而言,定義的時候不賦值,則是一個不確定的值。
4、雖然靜態變量在函數調用結束後仍然存在,但是其他函數不能引用。

二、用extern聲明外部變量。
用extern聲明外部變量,是爲了擴展外部變量的作用範圍。比如一個程序能由多個源程序文件組成。如果一個程序中需要引用另外一個文件中已經定義的外部變量,就需要使用extern來聲明。
正確的做法是在一個文件中定義外部變量,而在另外一個文件中使用extern對該變量作外部變量聲明。
一個文件中: int abc;
另外一個文件中: extern abc;

例子:
用extern將外部變量的作用域擴展到其他文件:
文件1:
//用extern將外部變量的作用域擴展到其他文件中
#include
#include
#include
unsigned int array[10];
void fillarray();
void init_ser()
{
SCON=0X50;
TMOD|=0X20;
TH1=0XF3;
TR1=1;
TI=1;
}
void main()
{
unsigned int i;
init_ser();
fillarray();
for(i=0;i<10;i++)
{
printf("array[%d]=%d\n",i,array[i]);
}
for(;;){;}
}

文件2:
extern int array[10];
void fillarray()
{
unsigned char i;
for(i=0;i<10;i++)
{
array[i]=i;
}
}

 

在單片機c語言中變量的空間分配幾個方法

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

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

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

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區。


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