单片机的存储器

【转】http://blog.sina.com.cn/s/blog_68f9692f0100jwr4.html

今天同事遇到一个内存溢出的问题,在帮忙解决过程中发现自己对这些东西还没有彻底弄清楚,就搜集了一些资料整理一下,受益匪浅。以前也记过笔记,但是时间一长又忘了,还是放在这里好了。

一.51的存储器结构

笼统来说单片机片内存储器分为CODE区和data区,cpu从code区读取指令,对data区的数据进行运算处理。前者在程序运行中为只读,一般为FLASH,用来放置程序代码和一些只读的数据(如字模表之类),后者可以随机读写,用来存放程序运行中的临时数据,如局部变量或全局变量,全局变量一直占用着RAM内存,而局部变量在使用完后会自动清除RAM空间。当然在片外,可以外扩FLASH和RAM(此时称为XRAM,因为寻址方式不一样),外扩的大小与单片机寻址能力有关。有的单片机将外扩RAM封装在片内,如AT89C51,所以有了所谓的片内XRAM。

二.变量的存储模式

变量是一种在程序执行过程中能不断变化的量。它有数据类型、存储类型、存储器模式和有效范围四种属性。C语言规定每个变量必须有一个标识符作为变量名,在使用一个变量前,必须先对变量进行定义,指出其数据类型和存储模式。以便编译系统为其分配存储单元。在C51中对变量进行定义的格式如下: 
   [存储类型] 数据类型 [存储器类型] 变量名表,如auto char data i;

存储类型指明变量的存储区域,而变量的存储类型和变量在程序中说明的位置决定了它的作用范围。存储类型含义与C语言相同。存储类型有四种:auto型、extern型、static型、register型,缺省为auto型(看来我一直都是凹凸型啊)。其区别如下:

auto:自动变量。存储在内存的堆栈区,属于临时性存储变量,并不长期占用内存,可以被多次覆盖。
    register:寄存器变量。register与auto一样属于自动类别。区别在于register的值保存在CPU的寄存器中。计算机中只有寄存器中的数据才能直接参与运算,而一般变量是放在内存中的,变量参加运算是,需要先把变量从内存中取到寄存器中,然后计算。所以一般把使用最频繁的变量定义成register变量。register变量只能在函数中定义,并只能是int和char型。
    static:静态变量。声明静态变量的,也就是C语言中的私有成员.如果在一个函数中声明一个静态变量,静态变量的空间不在栈里面,而是存储在静态空间里,这个函数结束后,静态变量的值依旧存在,内存不会收会此变量占用的内存空间,而是等整个程序都结果后才收回静态变量空间。

extern:外部类型。extern用来声明外部变量,可以用于此程序外的程序中(可在两个C文件间交叉使用),类型要一致。变量在数据运行时被分配了一定的内存空间,该空间在整个运行程序中,只要程序存在,自始自终都被该变量使用,即其值始终不变。

数据类型就不用多说了,bit,byte,char什么的。

存储器类型与单片机的寻址方式有关影响程序的执行效率。下表是传统C51的存储器类型,不同单片机类型有所差别。


空间名称

地址范围

说明
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,应该只能以BANK为单位读写。

简单解释一下:

data:  低128字节,可直接寻址,可以用acc直接读写的,速度最快,生成的代码也最小。

bdata:16字节位寻址区(当然也可以按字节寻址),一般很小。

idata: 固定指前面0x00-0xff的256个RAM,其中前128和data的128完全相同,只是因为访问的方式不同。idata是用类似C中的指针方式访问的。汇编中的语句为:mox ACC,@Rx。

xdata: 外部扩展RAM,一般指外部0x0000-0xffff空间,有的集成于片内,用DPTR访问。

pdata: 外部扩展RAM的低256个字节,用movx ACC,@Rx读写。这个比较特殊,而且C51好象有对此BUG,建议少用。

定义变量时如果省略了“存储器类型”,则按编译模式SAMLL, COMPACT,LARGE所规定的默认存储器类型确定变量的存储区域,(#pragma 预编译命令,可以指定函数的默认存储器模式。)C51编译器的三种存储器模式(默认的存储器类型)对变量的影响如下:
   1. SMALL:变量被定义在8051单片机的内部数据存储器中,因此对这种变量的访问速度最快。另外,所有的对象,包括堆栈,都必须嵌入内部数据存储器,而堆栈的长度是很重要的,实际栈长取决于不同函数的嵌套深度。
   2. COMPACT:变量被定义在分页外部数据存储器中,外部数据段的长度可达256字节。这时对变量的访问是通过寄存器间接寻址(MOVX@Ri)进行的,堆栈位于8051单片机内部数据存储器中。采用这种编译模式时,变量的高8位地址由P2口确定。
   3. LARGE:变量被定义在外部数据存储器中(最大可达64K字节),使用数据指针DPTR来间接访问变量,访问数据速度慢,增加程序代码的长度。
   存储器模式决定了缺省变量的存储空间,而访问各空间变量的汇编代码的繁简程度决定了代码率的高低。

例如:一个整形变量i,如放于内存18H、19H空间,则++ i的操作编译成四条语句:
INC  0x19
MOV  A,0x19
JNZ  0x272D
INC  0x18
0x272D:
   而如果放于外存空间0000H、0001H则++i的操作编译成九条语句:
MOV DPTR,0001
MOVX A,@ DPTR
INC A
MOVX @ DPTR,A
JNz  #5
MOV OPTR,#0000
MOVX A,@DPTR
INC A
MOVX @ DPTR,A

对于函数而言,一个函数的存储器模式确定了函数的参数和局部变量在内存中的地址空间,规则与变量定义一致。在定义一个函数时可以明确制定该函数的存储器模式,一般的形式如下:
              函数类型   函数名(形式参数表)  [存储器模式]
    其中的存储器模式是选项,未说明时则按该函数编译时的默认存储器模式处理。
    例如:
#pragma large   
int func1(int a1,  int a2) small  

 int c;
 ……
 ……
}

int func2(int b1,  int b2)     
{
 int c;
 ……
 ……
}
#pragma 预编译命令,可以指定函数的默认存储器模式。C51允许采用存储器的混合模式编程,充分利用单片机中有限的存储器空间,同时还可以加快程序运行的速度。
三、网友的一点经验

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

四.关于STARTUP.A51

   用C语言编程时,开机时执行的代码并非是从main()函数的第一句语句开始的,在main()函数的第一句语句执行前要先执行一段起始代码,这段起始代码的源程序名为STARTUP.A51。其作用可以看看源代码,简单来说就是进行变量的初始化,设置SP指针、堆栈空间等。如果考虑冷复位和热复位时的数据保存问题可以通过修改它实现(貌似不推荐)。

BTW,冷复位用英文来表示是Restart,热复位用英文来表示是Reset。我们把单片机从没加电到加上电源,而自动产生的复位称为冷复位;单片机在已经通电的情况下,给它一个复位信号,称为热复位。冷复位会使单片机的特殊功能寄存器和数据存储器的内容都改变;而热复位只是特殊功能寄存器的内容改变而单片机的内部数据存储器的内容不变。

六.关于中断

   我都困了,坚持做完吧,难得这么整理一回。
    中断时很好用的,与查询方式分半边天。C51编译器支持在C语言程序中直接编写51单片机的中断服务程序,C51编译对函数定义进行了扩展,增加了一个关键字interrupt,interrupt是函数定义时的一个选项,加上它函数将函数定义成中断服务函数。

函数类型 函数名(形式参数表) [interrupt n][using n]

interrupt 后面的n为中断号,n的取值范围为0~31,编译器从8n+3处产生中断向量。C51编译器还扩展了一个关键字using,专门用来选择单片机的寄存器组,缺省时由编译器选择一个寄存器组作为绝对寄存器组访问。

对于这个using,多说几句,普通函数也可以用。任何时候,单片机只能用到四组寄存器中的一组,一般情况下keil来自动分配,用USING来选择其中的一组,目的是提高效率减少出入栈次数。
   编写8051单片机中断函数时应遵循以下规则
    (l)中断函数不能进行参数传递,如果中断函数中包含任何参数声明都将导致编译出错。
    (2)中断函数没有返回值,如果企图定义一个返回值将得到不正确的结果。因此建议在定义中断函数时将其定义为void类型,以明确说明没有返回值。
    (3)在任何情况下都不能直接调用中断函数,否则会产生编译错误。因为中断函数的返回是由8051单片机指令RETI完成的,RETI指令影响8051单片机的硬件中断系统。如果在没有实际中断调求的情况下直接调用中断函数,RETI指令的操作结果会产生一个致命的错误。
    (4)如果中断函数中用到浮点运算,必须保存浮点寄存器的状态,当没有其它程序执行浮点运算时可以不保存。
    (5)如果在中断函数中调用了其它函数,则被调用函数所使用的寄存器组必须与中断函数相同。用户必须保证按要求使用相同的寄存器组。否则会产生不正确的结果,这一点必须引起足够的注意。如果定义中断函数时没有使用using选项,则由编译器选择一个寄存器组作绝对寄存器组访问。
七、关于volatile

   顺眼看到,就贴过来看看。

volatile是类型修饰符,影响编译器编译的结果,一般这个修饰符用来告知编译器,被修饰的变量是个“易变的”变量(volatile的本意是“易变的”),与volatile变量有关的运算,不要进行编译优化,以免出错。(加volatile关键字的变量有关的运算,将不进行编译优化。)
例如:volatile int i=10;
   int j = i;
   int k = i;
(1)volatile 告诉编译器i是随时可能发生变化的,每次使用它的时候必须从i的地址中读取,因而编译器生成的可执行码会重新从i的地址读取数据放在k中。
(2)而优化做法是,由于编译器发现两次从i读数据的代码之间的代码没有对i进行过操作,它会自动把上次读的数据放在k中。而不是重新从i里面读。这样以来,如果i是一个寄存器变量或者表示一个端口数据就容易出错,所以说volatile可以保证对特殊地址的稳定访问,不会出错。
常用情况:
1、中断服务程序中修改的供其它程序检测的变量需要加volatile;
2、多任务环境下各任务间共享的标志应该加volatile;
3、存储器映射的硬件寄存器通常也要加volatile说明,因为每次对它的读写都可能由不同意义。

时间: 2024-10-23 04:36:22

单片机的存储器的相关文章

单片机(simple chip microcomputer)概论、51单片机、MCS-51单片机、STM32、ARM

catalogue 1. 单片机概论 2. 51单片机 3. MCS-51单片机 4. QX-MINI51开发板(STC89C52芯片) 5. STM32单片机 6. stm32f103zet6开发板实验 1. 单片机概论 0x1: 仿真器 单片机应用系统的软硬件调试需要专门的开发工具,称为"单片机开发系统"或"仿真机".常用的开发方式是把开发系统(例如PC)中的CPU和RAM暂时出借给用户系统,利用开发系统对用户系统的软硬件进行调试(又称仿真),然后将调试好的程序

[学习笔记]15个QA让你快速入门51单片机开发

一.C语言相关... 1 Q1:sbit与sfr代表是什么?有什么作用?... 1 Q2:#define OSC_FREQ  22118400L这句宏命令里的“L”是什么意思?... 1 Q3:我粘贴了别人的代码,怎么发现没有unit这个类型?... 1 Q4:为什么好多变量都是char类型?它不是字符类型吗?怎么可以用来计数?    1 Q4.1:51单片机中的char,int,long,float,double各占多少个字节,取值范围多大?    1 Q5:unsigned char dat

[stm32参考手册] 1、Introduction

STM32F101xx, STM32F102xx, STM32F103xx, STM32F105xxand STM32F107xx advanced ARM-based 32-bit MCUs 本资料覆盖范围: 这本资料提供完整的关于上述系列STM单片机的存储器和外设的使用方法.在整个文档中(除非特别说明)是把这几类单片机归于STM32F10XXX系列的. 注:STM32F10XXX是一个包含的各种不同存储器尺寸,不同封装和不同外设的微处理器家族. 其他相关资料引导: 关于订货编号.电气和物理性

智能家居系统-硬件设计

1. 主控板设计 采用Altium Designer 09绘制系统PCB图, PCB图通过了设计规则检查(DRC),保证电气连接的正确性.PCB图采用手动布线的方式,在布线布局方面进行了优化.PCB图如图4-16所示,3D图如图4-17所示.焊接完成的PCB实物图如图4-18所示. 图4-16 中央控制器PCB图 3D视图 图1-17  中央控制器PCB的3D图 2. 学习型红外遥控器设计 学习型遥控器(万遥)的功能主要分为学习和发送两个部分.在学习的过程中,万遥接收到用户的家电遥控器所发送过来

硬件的自我修养

<怎样销售你自己>这本书的开头这样写到"决定一本书价值的诸多因素中,什么是最重要的?""要看作者,看他是否言之有物,而且又有能力表述完整,最重要的是作者本人是否就是运用书中道理的活典范?"所以先往自己脸上贴金,说一下自己的经历:大四上学期我找到杭州这份研究所的工作,05年3月份我就到研究所实习,后来一直干到2008年合同到期.第一年优秀新员工,第二年优秀团干部,第三年先进工作者.写过2篇所谓的科技报告,参加过995工程.863课题.也爬过舰桥.钻过潜艇,

深入了解DSP与ARM的区别与联系

深入了解DSP与ARM的区别与联系 2011-09-30 12:49:43|  分类: 嵌入式の半入其室 |  标签:体系结构  |举报|字号 订阅 下载LOFTER我的照片书  | 这些天正准备找工作的事,对于一些理论上的,或者说表面上的知识需要梳理下,所以有空整理了这篇简陋的比较,权当从另一个侧面理解下这两款主流处理器的特点了吧! DSP: DSP(digital singnal processor)是一种独特的微处理器,有自己的完整指令系统,是以数字信号来处理大量信息的器件.一个数字信号处

ARM、8051、AVR、MSP430、Coldfire、DSP、FPGA七种体系比较区别

概述 我以为这样比没有意义,做嵌入式系统最大特征是“嵌入”二字,也就是说你的控制系统是嵌入于你的控制对象之中,所以首先是服从于对象的需求和特征,脱离对象空论谁好谁坏有何依据? 每个MCU都有其存在的价值,每个使用者的选择都有其道理,AVR开始时是以单时钟周期指令为卖点,相对于当时 12个时钟的经典51确实有优势,而且基于CMOS的特征,时钟越高功耗越大,所以它在能耗上似乎明显占优. 可随着技术的改进,51现在已经早就有了4时钟周期,`2时钟周期乃至单时钟周期的芯片了,此时AVR的速度优势已不存在

KEIL、uVision、RealView、MDK、KEIL C51之间比较

KEIL uVision,KEIL MDK,KEIL For ARM,RealView MDK,KEIL C51,KEIL C166,KEIL C251 从接触MCS-51单片机开始,我们就知道有一个叫KEIL的软件.在开发单片机时,使用的是C语言或者汇编语言,我们知道,这两种语言都不能直接灌到单片机里面,执不执行暂且不说,光是代码的体积,就足以撑破整个单片机.所以,我们需要一个软件,把C语言或者汇编语言编译生成单片机可执行的二进制代码,而且它的体积也非常的小,足够存放在单片机的存储器里面. K

Keil、uVision、RealView、MDK、Keil C51之间的区别

Keil.uVision.RealView.MDK.Keil C51之间的区别 我们要区别的概念:KEIL uVision,KEIL MDK,KEIL For ARM,RealView MDK,KEIL C51,KEIL C166,KEIL C251 从接触MCS-51单片机开始,我们就知道有一个叫KEIL的软件.在开发单片机时,使用的是C语言或者汇编语言,我们知道,这两种语言都不能直接烧写到单片机里面,执不执行暂且不说,光是代码的体积,就足以撑破整个单片机. 所以,我们需要一个软件,把C语言或