51单片机C语言 几个数组定义的区别

这样三个句子
const uchar a[12]={...........};
char idata time[4]={...........};
uchar code table[20]={...........};

const idata code 这三个是什么作用?句子是什么意思?

const 表示本数组不可修改 数组为常量数组

code 表示本数组生成后是在ROM区中 同样不可修改

idata 表示数组生成后在在0x00~0xff的256个RAM中,使用指针寻址

具体的参考下面

data,bdata,idata,pdata,xdata,code存储类型与存储区

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空间了又增加了指针类型的分辨值。
温馨提示:内容为网友见解,仅供参考
第1个回答  推荐于2017-09-07

    CONST:通常类型是指使用类型修饰符const说明的类型,常类型的变量或对象的值是不能被更新的。(当然,我们可以偷梁换柱进行更新)可以定义const常量,具有不可变性。

    code 表示本数组生成后是在ROM区中 同样不可修改。

    idata 表示数组生成后在在0x00~0xff的256个RAM中,使用指针寻址。

    data,bdata,idata,pdata,xdata,code存储类型与存储区
    bit是在内部数据存储空间中 20H .. 2FH 区域中一个位的地址,或者 8051 位可寻址 SFR 的一个位地址。
    code是在 0000H .. 0FFFFH 之间的一个代码地址。
    data是在 0 到 127 之间的一个数据存储器地址,或者在 128 .. 255 范围内的一个特殊功能寄存器(SFR)地址。
    idata是 0 to 255 范围内的一个 idata 存储器地址。
    xdata 是 0 to 65535 范围内的一个 xdata 存储器地址。

第2个回答  2010-04-09
推崇一楼的答案,详细!

51单片机C语言 几个数组定义的区别
const 表示本数组不可修改 数组为常量数组 code 表示本数组生成后是在ROM区中 同样不可修改 idata 表示数组生成后在在0x00~0xff的256个RAM中,使用指针寻址 具体的参考下面 data,bdata,idata,pdata,xdata,code存储类型与存储区 bit是在内部数据存储空间中 20H .. 2FH 区域中一个位的地址,或者 8...

单片机C语言中: 定义 int idata i;和 idata int i 的区别
idata和xdata的地址空间范围不同,以51单片机举例,idata一般就是指单片机内部128字节的RAM中,而xdata是指数据存放在外扩的RAM芯片中,导致了地址存取范围和寻址方式完全不同。从汇编语言的角度来说,idata使用MOV指令即可,而xdata必须使用MOVX指令才能寻址。C51中变量的存储器类型有code,data,idata,xdata...

51单片机中 C语言 P2^3与P2.3的区别
P2.3 是 51单片机的一个引脚。而在 51 单片机的C语言中,根本就没有 P2.3。P2^3,就是 C51 语言和51单片机之间的桥梁。。

51单片机c语言程序
答:楼主先搞清C51中定义数据的类型。char型 8位(-127到128);int型16位-32727到+32728(两个字节)unsigned char是8位但从0到255,unsignde int 16位从0到65535;楼主程序中的TH0\/TL0是8位uchar型;tmp是16位uint型。tmp=TL0|(TH0<<8);这句在执行时先进行类型转换,把TL0\/TH0暂时转换成...

51单片机c语言编程中,有没有只定义p2.1到p2.6的写法?不要一个个的定义...
对于51单片机而言是不能一次性定义单独的一个IO口的某些位的,Keil里面没有提供相应的处理方法,有些单片机的C语言有此功能,用到了联合和结构的方式。比如在瑞萨的单片机C语言上可以这样定义:union { \/* EBR1 *\/ unsigned char BYTE; \/* Byte Access *\/ struct { \/* Bit...

51单片机C语言编程,是不是不能用位定义sbit来定义数组呢?我试过,会...
没法定义bit数组,bit的结构体不能定义,如果是做标志的话可以这样用 unsigned int flag[100]=0;\/\/定义1600个标志void SetFlag(int SetBit)\/\/标志置1{ flag[SetBit>>16]|=1<<(SetBit&0x0f);}void ClrFlag(int ClrBit)\/\/标志清零{ flag[ClrBit>>16]&=~(1<<(ClrBit&0x0f));...

c51单片机,如何定义可变长数组?
用int定义长数组是16位的,没有问题的,可以定义的,也没有限制的。下面的程序是定义了30元素的。用单步执行来查看数组的值,用P1口显示高8位,用P0显示低8位,即可以赋值,也可以输出值的,没有问题,这个程序测试通过的。include <reg52.h> int number[30];unsigned char i;void main(){ for...

51单片机的c语言与传统的c语言有什么区别
区别还是有的,51单片机的C语言面向的是单片机,所以硬件方面的函数和定义更多,很多相同名称的函数其实意思是不完全一样的。比如printf函数,传统的C语言是格式化输出到显示器显示,而单片机则是通过串口将数据格式化传送出去。

51单片机c语言编程中sbit和define的区别
1. `sbit` 用于在C语言中直接访问特定端口的位地址。它用于对51单片机等微控制器的I\/O端口中的特定位进行操作。使用`sbit`定义的位地址在编译时会被处理器特定的编译器识别,确保了代码针对特定硬件的优化。2. `define` 是C语言的预处理器指令,用于定义宏常量或进行文本替换。它广泛用于定义常量、...

51单片机的优先级怎么用C语言怎么写?
... ;其他主程序TIMER0: ;真正的中断服务程序 ...RETI ;返回到主程序而用C语言就更简单了,把中断服务程序就是一个函数,不过这个函数加个interrupt的关键字来修饰就是了。而interrupt后面加的是个中断的中断号。由于复位中断的发生是不可逆的,发生后不可能再返回到发生中断之前。所以...

相似回答