data xdata btata 等区别

本文详细介绍了C51编程中涉及的存储类型,如data、bdata、idata、xdata和code,以及它们与存储区的关系。讨论了指针类型和存储区的结合使用,通过实例展示了不同声明方式在编译后的汇编代码,强调了正确使用存储类型的重要性,以提高代码效率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

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) (256 BYTE/页)
    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中C51 User'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种方式。

在MCU(微控制器,如8051系列)中,`data`、`idata` 和 `xdata` 是用于描述不同存储类型的关键字,它们定义了变量在内存中的存储位置和访问方式。这些关键字主要用于Keil C等嵌入式C编译器中,以便开发者能够更精确地控制内存分配和访问效率。 ### `data` 存储类型 `data` 类型的变量存储在MCU的内部RAM低128字节(0x00–0x7F)区域中,这部分内存可以被直接寻址,因此访问速度非常快。此区域通常用于频繁访问的变量,例如循环计数器或中断服务程序中使用的变量[^1]。 ```c unsigned char data var1; // 变量 var1 存储在内部RAM低128字节 ``` ### `idata` 存储类型 `idata` 类型的变量也存储在内部RAM中,但其地址范围是0x00–0xFF,包括低128字节和高128字节(即SFR区域之后的RAM区域)。这部分内存使用间接寻址方式进行访问,因此访问速度比`data`稍慢,但比`xdata`快。`idata` 通常用于需要较多内部RAM变量但不需要最高访问速度的场景[^1]。 ```c unsigned char idata var2; // 变量 var2 使用间接寻址访问 ``` ### `xdata` 存储类型 `xdata` 类型的变量存储在外部RAM中,地址范围通常是0x0000–0xFFFF,这需要通过外部总线接口进行访问。由于访问外部RAM需要额外的指令周期,因此`xdata`类型的变量访问速度最慢。`xdata` 适用于存储量大但访问频率较低的数据[^1]。 ```c unsigned char xdata var3; // 变量 var3 存储在外部RAM中 ``` ### 存储类型对比总结 | 存储类型 | 存储位置 | 地址范围 | 访问方式 | 访问速度 | 用途 | |----------|-----------|------------|-------------|------------|------| | `data` | 内部RAM | 0x00–0x7F | 直接寻址 | 快 | 高频访问变量 | | `idata` | 内部RAM | 0x00–0xFF | 间接寻址 | 中 | 中等访问频率变量 | | `xdata` | 外部RAM | 0x0000–0xFFFF| 外部总线访问 | 慢 | 大容量数据存储 | ### 使用建议 - **优先使用`data`**:对于需要快速访问的变量,如中断处理程序中使用的变量,应优先使用`data`类型。 - **合理使用`idata`**:当内部RAM需求较大时,可以使用`idata`来扩展内部RAM的使用,但需接受稍慢的访问速度。 - **`xdata`用于大容量数据**:当数据量较大且访问频率较低时,使用`xdata`可以节省内部RAM资源,但会牺牲访问性能。 ### 示例代码 ```c #include <reg51.h> // 定义不同存储类型的变量 unsigned char data fastVar; // 内部RAM低128字节 unsigned char idata mediumVar; // 内部RAM高128字节 unsigned char xdata slowVar; // 外部RAM void main() { fastVar = 0x10; // 快速访问 mediumVar = 0x20; // 间接访问 slowVar = 0x30; // 外部访问 while (1) { // 主循环 } } ``` ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值