当前,单片机种类很多,且实际工作中仅应用一种单片机也是不现实的,必得对常用的几种单片机有所了解。

尽管现在单片机编程大多使用C语言,但必得对单片机的内核结构、存储结构及指令集有一定的了解,才有可能写出优秀的程序代码。对于单片机指令的学习,寻址方式的学习是其中的一个重点和难点,寻址方式的正确理解不仅对汇编编程至关重要,而且有助于对于单片机内核结构(如RISC和CISC的区别)、存储结构的更深刻理解。

但是,不同单片机都提供了一些不同的寻址方式,且即使同样的寻址方式在不同的单片机中也有不同的名称,使得寻址方式显得混乱,不一致,不易理解。

不过,经过仔细的对比、学习和分析,我发现,其实所有的寻址方式,都可以归为以下六类:

1)立即寻址

2)无址寻址

3)寄存器直接寻址

4)寄存器间接寻址

5)内存直接寻址

6)内存间接寻址

下面对以上六类指令一一分解:

1、立即寻址,即在指令中直接给出实际的操作数数值,如MOV R1,#0

2、无址寻址,就是说指令中根本没有给出操作数或操作数地址,其原因可能是以下两者之一:其一,本指令确实不需要操作数,如NOP指令;其二,本指令本身就是专门为某一个操作数制造的,它不可能用来操作其他的操作数,如CLRA指令(由其名称可知,它用来清零累加器A)。

3、寄存器直接寻址,这是一种十分重要,十分常见,也十分易于理解、易于掌握和应用的寻址方式,就是在指令中直接给出寄存器编号(在汇编中,给出的是寄存器的标号,也可以视作寄存器的名称)作为操作数的寻址方式,而待操作的数据就存在此寄存器中。如指令MOV R1,R2就是一条双操作数都采用寄存器直接寻址的指令。

另外,这种寻址方式在一些处理器构架中又会有一些细微变化,如在ARM中由于大量使用寄存器直接寻址方式,为了增强此寻址方式的威力,ARM处理器中的寄存器直接寻址加入了移位功能,即在实际指令执行中的数据不是简单的寄存器中直接存储的数据,而是此数据经过移位后得到的数据。

4、寄存器间接寻址,如同寄存器直接寻址,都是在指令编码中给出寄存器编号(在汇编中表现为寄存器名称);但与寄存器直接寻址不同的是,此处寄存中存储的并非操作数本身,而是操作数在内存中的位置,即操作数的地址(相当于C语言中的指针),在内存中此地址处存储的才是真正的操作数。所以,CPU要获取此操作数,在微观上需要进行两次数据获取操作,第一次从寄存器中获取操作数的地址,第二此从刚才获取的地址处在内存中获取真正的操作数。在多数处理器中,指令使用寄存器间接寻址比使用寄存器直接寻址运行要耗费更多的指令周期。如指令 MOV R1,@R2将R1中的数据复制到R2中地址所指向的内存中。

寄存器间接寻址是变种最多、扩展最多的寻址方式,正因如此,他也是处理器中最为灵活、最为强大、最为难于掌握的寻址方式。不仅不同处理器对它进行了不同的扩展,而且即使对于完全相同的扩展,不同汇编中也可能给出相差极大的书写格式,这就更加导致了寄存器间接寻址难于理解、难于掌握。

其扩展主要有以下形式:1)自增、自减型,@R1++、@R1--、@(--R1)、@(++R1),看到它们的书写形式大家应该也大体明白它们的意思了,当然具体的汇编书写格式那就根据不同的处理器和汇编器变得五花八门了,比如有的不用@号而用中括号,有的只用一个加减号而不是两个,有的只提供了四种变体中的一种或几种,情况不一而足,但其实质内容却大体一致。2)索引型(或叫偏移型),就是指寄存器中存储的地址还不是操作数的最终地址,须得对此地址加一个数才是最终的操作数地址。这种情况就更加复杂了,其一,书写格式更加纷乱,如我就见过X(R1)、[R1,X]、(R1+X)等多种形式;其二,偏移量X本身也涉及寻址方式这个问题,最常用的当然是立即寻址和寄存器直接寻址两种,但也不排除有更加复杂的寄存器间接寻址这种方式的使用,要是这个X本身要是也能使用寄存器间接寻址的变种——索引寻址,那可就真是热闹了(试想一下,这不就是迭代了吗)。

5、内存直接寻址,就是在指令中直接给出操作数的地址——存储于内存中的位置,表现在汇编中,为了区分立即数和内存地址,一般对地址加&或$前缀。如 MOV R1,$100

这个内存地址可以可以分两种情况给出:其一,直接给出完整的地址;其二,仅给出一个偏移量,将此偏移量加上PC才能得到操作数的地址。其中,前者一般叫做绝对寻址,后者一般叫做相对寻址(当然,有些汇编中可能并不这样称呼),二者很容易理解,也很容易区分。当然,二者也是各有优劣:绝对寻址简化了执行电路,因为他无需进行PC和偏移量的加法运算;而相对寻址则可精简编码长度,从而节省程序存储器消耗,当然,还有一个更加重要的好处,那就是可以利用这个特性编写“可搬移”的代码。因为使用相对地址,那么把程序整段搬移到其他的地址空间去,程序也能够正确的执行而不致出差。

对于同时提供了绝对寻址和相对寻址的CPU,在其汇编语法层面就需要给出区别两种寻址方式的语法,由于不同汇编给出的解决方法不同,这里仅以MSP430编程手册中给出的汇编语法为例进行介绍,以方便理解:

相对寻址:MOV ADDR,R4

绝对寻址:MOV &ADDR,R4

可见,这里是通过一个“&”符合进行绝对寻址和相对寻址的区分的。

另外,还有一个不大不小的问题需要注意,很多时候,使用内存直接寻址的汇编中并不直接给出地址数值,而是给出“标号,即label”,汇编器自己把这个标号用实际的地址值进行替代,所以就免除了地址的手工计算的麻烦。

6、内存间接寻址,参考寄存器直接寻址与寄存器间接寻址的异同,我们可以通过上面讲到内存直接寻址来推测出内存间接寻址的内容和概念。但是,鉴于在嵌入式处理器和单片机中一般都不使用内存间接寻址方式,所以,这里也就不再展开谈了。

另外,应该清楚,寻址方式并不是处理器设计者想怎么设计就怎么设计的,它很大程度上是由处理器内核结构和存储结构所决定的。一般来说,立即寻址和无址寻址在所有的处理器或单片机中都是存在的;而RISC构架的处理器中一般大量使用寄存器直接寻址,而寄存器间接寻址和内存直接寻址一般只用于加载(LDR)、存储(STR)指令之中,内存间接寻址则在RISC中基本不使用;而对于CISC架构的处理器,不仅大量使用寄存器直接寻址,而且大量使用灵活多变的寄存器间接寻址和内存直接寻址,甚至于会提供强大的内存间接寻址。所以,CISC构架处理器的指令一般功能更丰富、更强大,可以用比RISC技术更少的指令完成相同的功能,但代价是必须增加更多的地址解析译码电路,增加了芯片体积和功耗。

当然,我学单片机时间也不长,水平一般,也没有了解过太多的单片机,只是就自己了解过的ARM、51、AVR、PIC和MSP430做一个小小总结,希望能帮助初学者更快的学习和理解单片机指令。但是能力和见识有限,未免以偏概全,所以也希望功力深厚、见识广博的老前辈能够指正错误,或添加其他未提到的寻址方式。当然,若是那位有更好的寻址方式归纳总结方法分享,那将是更好了。

52单片机C语言如何用间接寻址,单片机要如何寻址?相关推荐

  1. 51单片机c语言试题及答案,单片机C语言期末考试题..

    单片机C语言期末考试题..,单片机的c语言应用程序设计,单片机c语言编程,51单片机c语言程序100例,单片机c语言,单片机c语言编程思想,c语言开发单片机,c语言单片机程序,新概念51单片机c语言教 ...

  2. 单片机c语言 i%3c%3c1,单片机C语言作业及上机习题及答案

    <单片机C语言作业及上机习题及答案>由会员分享,可在线阅读,更多相关<单片机C语言作业及上机习题及答案(37页珍藏版)>请在人人文库网上搜索. 1.第一次课熟悉winTC编译环 ...

  3. 单片机c语言idata什么意思,单片机C语言中的data,idata,xdata,pdata,code

    从数据存储类型来说,8051系列有片内.片外程序存储器,片内.片外数据存储器,片内程序存储器还分直接寻址区和间接寻址类型,分别对应code.data.xdata.idata以及根据51系列特点而设定的 ...

  4. 51单片机c语言工作手册,51单片机C语言编程手册

    51单片机C语言编程手册 1 第一章 介绍 这是一本关于Intel 80C51 以及广大的51 系列单片机的书这本书介绍给读者一些 新的技术使你的8051 工程和开发过程变得简单请注意这本书的目的可不 ...

  5. 51单片机c语言编程100,51单片机C语言编程100例.doc

    51单片机C语言编程100例.doc 目 录实例3用单片机控制第一个灯亮3实例4用单片机控制一个灯闪烁认识单片机的工作频率3实例5将 P1口状态分别送入P0.P2.P3口认识I/O口的引脚功能4实例6 ...

  6. 51单片机C语言code定义,51单片机数组的定义方法(code与data的作用)

    数组前不加"code"或"data",则默认将数组存放在程序存储器中: code 指定数据是存储在代码区,数据是在编程的时候跟代码一起写入代码存储器,运行过程中 ...

  7. 51单片机c语言试题及答案,单片机考试复习题及答案

    <单片机考试复习题及答案>由会员分享,可在线阅读,更多相关<单片机考试复习题及答案(14页珍藏版)>请在人人文库网上搜索. 1.选择题1.主频为12MHz的单片机他的机器周期为 ...

  8. 51单片机c语言内存使用,51单片机C语言编程之内存优化

    原标题:51单片机C语言编程之内存优化 最常见的是以下两种 ① 超过变量128后必须使用compact模式编译 实际的情况是只要内存占用量不超过 256.0 就可以用 small 模式编译 ② 128 ...

  9. 单片机c语言编程教学大纲,《单片机C语言编程》教学大纲

    <单片机C语言编程>教学大纲 课程代码:000002336 课程英文名称:Microcontroller C Programming Language 课程总学时:24 讲课:16 实验: ...

  10. pic单片机c语言存储器定义,PIC单片机C语言程序设计1 7.PDF

    PIC单片机C语言程序设计1 7 学电子跟我来FOLLOW ME PIC 单片机C 语言程序设计(1) ◆ 丁锦滔 编者按:为了帮助具有PI C 单片机汇编语言知识的技术人员或工程师,快速掌握利用C ...

最新文章

  1. Oracle rollup 关键字用法简介.
  2. IOS开发 REST请求 ASIHTTPRequest用法
  3. SAP S/4HANA生产订单释放后自动同步到MES系统
  4. Nginx服务器的开发团队
  5. 简单的窗体抖动托管c++
  6. web之JavaScript
  7. Linux 基本命令(三)--histroy 常用命令详解
  8. Redis详解(四)——配置文件详解
  9. Binary translation is incompatible with long mode on this platform
  10. 在xp中运行 .air 文件--Adobe AIR Runtime for Window
  11. 算法题04:分治法:求第K小元素(线性时间选择算法)
  12. 网络渗透基本思路及方法
  13. 数字图像处理实验MATLAB版+实验报告亲笔
  14. Oracle Wallet Manager
  15. 前端-微信、企业微信JS-SDK实时调试方法
  16. win7文件服务器代码,win7搭建文件服务器
  17. cold diffusion的个人理解
  18. Qt5.6.1如何使用qpf2字体
  19. oracle创建视图与生成唯一编号
  20. python将学号与成绩匹配_python输入学号输出成绩等级_python将百分制成绩转换为等级制输出...

热门文章

  1. 一、Windows许可证即将过期怎么办
  2. linux(所有版本)下安装有道词典
  3. 软件使用-如何卸载360安全客户端
  4. 树的深度优先和广度优先
  5. fins协议握手信号服务器响应,欧姆龙OMRON PLC之HostLink通讯协议(四)-FINS命令工作模式篇...
  6. python omron Fins -PLC和上位机socket通讯( tcy)
  7. 静态路由使用下一跳和出接口的区别,我猜你不知道这一点!
  8. 什么叫DMZ区?DMZ区它有什么作用?
  9. 多表连接查询和多次单表查询哪个效率高
  10. centos设置自动开关机