第四章汇编语言程序设计基础.docx
《第四章汇编语言程序设计基础.docx》由会员分享,可在线阅读,更多相关《第四章汇编语言程序设计基础.docx(23页珍藏版)》请在冰豆网上搜索。
第四章汇编语言程序设计基础
授课教师
授课班级
课的类型
授课时间
课时分配
授课地点
课题
汇编语言程序设计基础
教学目标
1、数据传送类指令
2、控制转移指令
3、延时程序的概念和程序设计
重 点
2、数据传送类指令
3、控制转移指令
4、延时程序的概念和程序设计
难 点
能够独立分析程序
教法
讲授法+多媒体形式
教具
多媒体
作业
1.课堂作业
教材P113页练习题
2.课下作业
打印片子一套
引入课题:
新课教学:
汇编语言是面向机器硬件的语言,要求程序设计者对MCS-51单片机具有很好的“软、硬结合”的功底。
介绍程序设计的基本知识及如何使用汇编语言来进行基本的程序设计。
4.1汇编语言程序设计概述
4.1.1机器语言、汇编语言和高级语言
用于程序设计的语言基本上分为3种:
机器语言、汇编语言和高级语言。
1.机器语言
二进制代码表示的指令、数字和符号简称为机器语言
不易懂,难记忆,易出错。
2.汇编语言
英文助记符表示的指令称为符号语言或汇编语言
将汇编语言程序转换成为二进制代码表示的机器语言程序称为汇编程序
经汇编程序“汇编(翻译)”得到的机器语言程序称为目标程序,原来的汇编语言程序称为源程序。
汇编语言特点:
面向机器的语言,程序设计员须对MCS-51的硬件有相当深入的了解。
助记符指令和机器指令一一对应,用汇编语言编写的程序效率高,占用存储空间小,运行速度快,用汇编语言能编写出最优化的程序。
能直接管理和控制硬件设备(功能部件),它能处理中断,也能直接访问存储器及I/O接口电路。
汇编语言和机器语言都脱离不开具体机器的硬件,均是面向“机器”的语言,缺乏通用性。
3.高级语言
不受具体机器的限制,使用了许多数学公式和数学计算上的习惯用语,非常擅长于科学计算。
常用的如BASIC、FORTRAN以及C语言等。
高级语言优点:
通用性强,直观、易懂、易学,可读性好。
使用C语言(C51)、PL/M语言来进行MCS-51的应用程序设计。
对于程序的空间和时间要求很高的场合,汇编语言仍是必不可缺的。
C语言和汇编语言混合编程
在很多需要直接控制硬件的应用场合,则更是非用汇编语言不可
使用汇编语言编程,是单片机程序设计的基本功之一
4.1.2汇编语言语句的种类和格式
两种基本类型:
指令语句和伪指令语句
(1)指令语句
已在第3章介绍
每一条指令语句在汇编时都产生一个指令代码——机器代码
(2)伪指令语句
是为汇编服务的。
在汇编时没有机器代码与之对应。
MCS-51的汇编语言的四分段格式如下:
标号字段操作码字段操作数字段注释字段
规则:
(1)标号字段和操作字码段之间要有冒号“:
”相隔;
(2)操作码字段和操作数字段间的分界符是空格;
(3)双操作数之间用逗号相隔;
(4)操作数字段和注释字段之间的分界符用分号“;”相隔。
操作码字段为必选项,其余各段为任选项。
例4-1下面是一段汇编语言程序的四分段书写格式
标号字段操作码字段操作数字段注释字段
START:
MOVA,#00H;0→A
MOVR1,#10;10→R1
MOVR2,#00000011B;3→R2
LOOP:
ADDA,R2;(A)+(R2)→A
DJNZR1,LOOP;R1内容减1不为零,则循环
NOP
HERE:
SJMPHERE
基本语法规则:
1.标号字段
是语句所在地址的标志符号
(1)标号后边必须跟以冒号“:
”
(2)由1~8个ASCII字符组成
(3)同一标号在一个程序中只能定义一次
(4)不能使用汇编语言已经定义的符号作为标号
2.操作码字段
是汇编语言指令中唯一不能空缺的部分。
汇编程序就是根据这一字段来生成机器代码的。
3.操作数字段
通常有单操作数、双操作数和无操作数三种情况。
如果是双操作数,则操作数之间,要以逗号隔开。
(1)十六进制、二进制和十进制形式的操作数表示
采用十六进制形式来表示,某些特殊场合才采用二进制或十进制的表示形式。
十六进制,后缀“H”。
二进制,后缀“B”。
十进制,后缀“D”,也可省略。
若十六进制的操作数以字符A~F中的某个开头时,则需在它前面加一个“0”,以便在汇编时把它和字符A~F区别开来。
(2)工作寄存器和特殊功能寄存器的表示
采用工作寄存器和特殊功能寄存器的代号来表示,也可用其地址来表示。
例如,累加器可用A(或Acc)表示。
也可用0E0H来表示,0E0H为累加器A的地址。
(3)美元符号$的使用
用于表示该转移指令操作码所在的地址。
例如,如下指令:
JNBF0,$
与如下指令是等价的:
HERE:
JNBF0,HERE
再如:
HERE:
SJMPHERE
可写为:
SJMP$
4.注释字段
必须以分号“;”开头,换行书写,但必须注意也要以分号“;”开头。
汇编时,注释字段不会产生机器代码。
4.1.3伪指令
在MCS-51汇编语言源程序中应有向汇编程序发出的指示信息,告诉它如何完成汇编工作,这是通过使用伪指令来实现的。
也称为汇编程序控制命令。
只有在汇编前的源程序中才有伪指令。
经过汇编得到目标程序(机器代码)后,伪指令已无存在的必要,所以“伪”体现在汇编时,伪指令没有相应的机器代码产生。
常用的伪指令:
1.ORG(ORiGin)汇编起始地址命令
在汇编语言源程序的开始,通常都用一条ORG伪指令来实现规定程序的起始地址。
如不用ORG规定,则汇编得到的目标程序将从0000H开始。
例如:
ORG2000H
START:
MOVA,#00H
┋
规定标号START代表地址为2000H开始。
在一个源程序中,可多次使用ORG指令,来规定不同的程序段的起始地址。
但是,地址必须由小到大排列,地址不能交叉、重叠。
例如:
ORG2000H
┇
ORG2500H
┇
ORG3000H
┇
2.END(ENDofassembly)汇编终止命令
汇编语言源程序的结束标志,用于终止源程序的汇编工作。
在整个源程序中只能有一条END命令,且位于程序的最后。
3.DB(DefineByte)定义字节命令
在程序存储器的连续单元中定义字节数据。
ORG2000H
DB30H,40H,24,“C”,“B”
汇编后:
(2000H)=30H
(2001H)=40H
(2002H)=18H(10进制数24)
(2003H)=43H(字符“C”的ASCII码)
(2004H)=42H(字符“B”的ASCII码)
DB功能是从指定单元开始定义(存储)若干个字节,10进制数自然转换成16进制数,字母按ASCII码存储。
4.DW(DefineWord)定义数据字命令
从指定的地址开始,在程序存储器的连续单元中定义16位的数据字。
例如:
ORG2000H
DW1246H,7BH,10
汇编后:
(2000H)=12H;第1个字
(2001H)=46H
(2002H)=00H;第2个字
(2003H)=7BH
(2004H)=00H;第3个字(2005H)=0AH
(2005H)=0AH
5.EQU(EQUate)赋值命令
用于给标号赋值。
赋值以后,其标号值在整个程序有效。
例如:
TESTEQU2000H
表示标号TEST=2000H,在汇编时,凡是遇到标号TEST时,均以2000H来代替。
4.1.4汇编语言程序设计步骤
(1)分析问题,确定算法
(2)根据算法,画出程序框图
(3)分配内存工作区及有关端口地址
(4)编写程序
养成在程序的适当位置上加上注释的好习惯。
(5)上机调试
编写完毕的程序,必须“汇编”成机器代码,才能调试和运行,调试与硬件有关程序还要借助于仿真开发工具并与硬件连接。
4.2汇编语言源程序的汇编
汇编语言源程序“翻译”成机器代码(指令代码)的过程称为“汇编”。
汇编可分为手工汇编和机器汇编两类:
4.2.1手工汇编
人工查表翻译指令。
但遇到的相对转移指令的偏移量的计算,要根据转移的目标地址计算偏移量,不但麻烦,且容易出错。
4.2.2机器汇编
用编辑软件进行源程序的编辑。
编辑完成后,生成一个ASCII码文件,扩展名为“.ASM”。
然后在微计算机上运行汇编程序,把汇编语言源程序翻译成机器代码。
交叉汇编—汇编后的机器代码是在另一台计算机(这里是单片机)上运行。
MCS-51单片机的应用程序的完成,应经过三个步骤;
(1)在微计算机上,运行编辑程序进行源程序的输入和编辑;
(2)对源程序进行交叉汇编得到机器代码;
(3)通过微计算机的串行口(或并行口)把机器代码传送到用户样机(或在线仿真器)进行程序的调试和运行。
第
(1)步,只需在微计算机上使用通用的编辑软件即可完成。
第
(2)步的交叉汇编所用的汇编程序可在购买单片机的仿真开发工具时,由厂商提供。
第(3)步骤的实现要借助于单片机仿真开发工具进行。
反汇编——分析现成产品的程序,要将二进制的机器代码语言程序翻译成汇编语言源程序。
例4-2下面是一段源程序的汇编结果,读者可通过查第3章的表3-3至表3-7,进行手工汇编,来验证下面的汇编结果是否正确。
见表4-1(P75)
4.3汇编语言实用程序设计
4.3.1汇编语言程序的基本结构形式
常
顺采用以下几种基本结构:
序结构、分支结构和循环结构,再加上广泛使用的子程序和中断服务子程序。
1.顺序结构
2.分支结构
程序中含有转移指令,
无条件分支,有条件分支。
有条件分支又分为:
单分支结构和多分支结构。
3.循环结构
4.子程序
5.中断服务子程序
4.3.2子程序的设计
一、子程序设计原则和应注意的问题
一种能完成某一特定任务的程序段。
其资源要为所有调用程序共享。
因此,子程序在结构上应具有独立性和通用性,在编写子程序时应注意以下问题:
1.子程序的第一条指令的地址称为子程序的入口地址。
该指令前必须有标号。
2.主程序调用子程序
两条子程序调用指令:
(1)绝对调用指令:
ACALLaddr11
(2)长调用指令:
LCALLaddr16
3.注意设置堆栈指针和现场保护
4.最后一条指令必须是RET指令
5.子程序可以嵌套,即子程序可以调用子程序
6.在子程序调用时,还要注意参数传递的问题
二、子程序的基本结构
MAIN:
┇;MAIN为主程序或调用程序标号
┇
LCALLSUB;调用子程序SUB
┇
┇
SUB:
PUSHPSW;现场保护
PUSHACC;
子程序处理程序段
POPACC;现场恢复
POPPSW;
RET;最后一条指令必须为RET
例4-3单字节有符号数的加减法子程序(自己阅读)
本例中参数传递是通过累加器A完成的,主程序将被转换的数送到A中,子程序将A中的有符号数求补后存于A中,主程序再将结果放回原来的单元。
例4-44位BCD码的减法程序(自己阅读)
主程序通过地址寄存器R0和R1将参加运算的BCD码的地址传递给子程序,子程序则通过累加器将差传递给主程序。
4.3.3查表程序设计
数据补偿、修正、计算、转换等各种功能,具有程序简单、执行速度快等优点。
查表就是根据自变量x,在表格中寻找y,使y=f(x)。
执行查表指令时,发出读程序存储器选通脉冲/PSEN。
在MCS-51的指令系统中,给用户提供了两条极为有用的查表指令:
⑴MOVCA,@A+DPTR
⑵MOVCA,@A+PC
指令“MOVCA,@A+DPTR”完成把A中的内容作为一个无符号数与DPTR中的内容相加,所得结果为某一程序
存储单元的地址,然后把该地址单元中的内容送到累加器A中。
指令“MOVCA,@A+PC”以PC作为基址寄存器,PC的内容和A的内容作为无符号数,相加后所得的数作为某一程序存储器单元的地址,根据地址取出程序存储器相应单元中的内容送到累加器A中。
指令执行完,PC的内容不发生变化,仍指向查表指令的下一条指令。
优点在于预处理较少且不影响其它特殊功能寄存器的值,所以不必保护其它特殊功能寄存器的原先值。
缺点在于该表格只能存放在这条指令的地址X3X2X1X0以下的00~FFH之中。
表格所在的程序空间受到了限制。
例4-5子程序的功能为:
根据累加器A中的数x(0~9之间)查x的平方表y,根据x的值查出相应的平方y。
x和y均为单字节数。
地址子程序
Y3Y2Y1Y0ADDA,#01H
Y3Y2Y1Y0+2MOVCA,@A+PC
Y3Y2Y1Y0+3RET
Y3Y2Y1Y0+4DB00H,01H,04H,09H,10H
DB19H,24H,31H,40H,51H
第1条指令ADDA,#01H的作用是加上偏移量,可以根据A的内容查出X对应的平方。
MOVCA,@+DPTR这条指令的应用范围较为广泛,一般情况下,大多使用该指令,使用该指令时不必计算偏移量,使用该指令的优点是表格可以设在64K程序存储器空间内的任何地方,而不像MOVCA,@A+PC那样只设在PC下面的256个单元中,使用较方便。
上面的程序可改成如下形式:
PUSHDPH;保存DPH
PUSHDPL;保存DPL
MOVDPTR,#TAB1
MOVCA,@A+DPTR
POPDPL;恢复DPL
POPDPH;恢复DPH
RET
TAB1:
DB00H,01H,04H,09H,10H
DB19H,24H,31H,40H,51H
例4-6在一个以MCS-51为核心的温度控制器中,温度传感器输出的电压与温度为非线性关系,传感器输出的电压已由A/D转换为10位二进制数。
根据测得的不同温度下的电压值数据构成一个表,表中放温度
值y,x为电压值数据。
设测得的电压值x放入R2R3中,根据电压值x,查找对应的温度值y,仍放入R2R3中。
本例的x和y均为双字节无符号数。
程序如下:
LTB2:
MOVDPTR,#TAB2
MOVA,R3
CLRC
RLCA
MOVR3,A
XCHA,R2
RLCA
XCHR2,A
ADDA,DPL;(R2R3)+(DPTR)
→(DPTR)
MOVDPL,A
MOVA,DPH
ADDCA,R2
MOVDPH,A
CLRA
MOVCA,@A+DPTR;查第一字节
MOVR2,A;第一字节存入R2中
CLRA
INCDPTR
MOVCA,@A+DPTR;查第二字节
MOVR3,A;第二字节存入R3中
RET
TAB2:
DW……;温度值表
例4-7设有一个巡回检测报警装置,需对16路输入进行检测,每路有一最大允许值,为双字节数。
运行时,需根据测量的路数,找出每路的最大允许值。
看输入值是否大于最大允许值,如大于就报警。
根据上述要求,编一个查表程序。
取路数为x(0x15),y为最大允许值,放在表格中。
设进入查表程序前,路数x已放于R2中,查表后最大值y放于R3R4中。
本例中的x为单字节数,y为双字节数。
查表程序如下:
TB3:
MOVA,R2
ADDA,R2;(R2)*2→(A)
MOVR3,A;保存指针
ADDA,#6;加偏移量
MOVCA,@A+PC;查第一字节
XCHA,R3
ADDA,#3
MOVCA,@A+PC;查第二字节
MOVR4,A
RET
TAB3:
DW1520,3721,42645,7580;最大值;表
DW3483,32657,883,9943
DW10000,40511,6758,8931
DW4468,5871,13284,27808
表格长度不能超过256个字节,且表格只能存放于MOVCA,@A+PC指令以下的256个单元中。
4.3.4关键字查找程序设计
顺序检索和对分检索
一、顺序检索
从第1项开始逐项顺序查找,判断所取数据是否与关键字相等。
例4-8从50个字节的无序表中查找一个关键字××”H。
ORG1000H
MOV30H,#××H;关键字××H送30H单元
MOVR1,#50;查找次数送R1
MOVA,#14;修正值送A
MOVDPTR,#TAB4;表首地址送DPTR
LOOP:
PUSHACC
MOVCA,@A+PC;查表结果送A
CJNEA,40H,LOOP1;(40H)不等于关键字则转LOOP1
MOVR2,DPH;已查到关键字,把该字
;的地址送R2,R3
MOVR3,DPL;
DONE:
RET
LOOP1:
POPACC;修正值弹出
INCA;A+1→A
INCDPTR;修改数据指针DPTR
DJNZR1,LOOP;R1≠0,未查完,继续查找
MOVR2,#00H;R1=0,清“0”R2和`R3
MOVR3,#00H;表中50个数已查完
AJMPDONE;从子程序返回
TAB4:
DB…,…,…;50个无序数据表
二、对分检索
前提:
检索的数据表已经排好序,如何进行数据的排序,将在本节稍后介绍。
方法:
取数据表中间位置的数与关键字进行比较,如相等,则查找到;如果所取的数大于关键字,则下次对分检索的范围是从数据区起点到本次取数。
如果取数小于关键字,则下次对分检索的范围是从本次取数数据区起点到数据区终点。
依此类推,逐渐缩小检索范围,减少次数,大大提高了查找速度。
4.3.5数据极值查找程序设计
在指定的数据区中找出最大值(或最小值)。
进行数值大小的比较,从这批数据中找出最大值(或最小值)并存于某一单元中。
例4-9片内RAM中存放一批数据,查找出最大值并存放于首地址中。
设R0中存首地址,R2中存放字节数,程序框图如图4-1所示。
程序如下:
MOVR2,n;n为要比较的数据字节数
MOVA,R0;存首地址指针
MOVR1,A
DECR2;
MOVA,@R1
LOOP:
MOVR3,A
DECR1
CLRC
SUBBA,@R1;两个数比较
JNCLOOP1;C=0,A中的数大,跳LOOP1
MOVA,@R1;C=1,则大数送A
SJMPLOOP2
LOOP1:
MOVA,R3
LOOP2:
DJNZR2,LOOP;是否比较结束?
MOV@R0,A;存最大数
RET
4.3.6数据排序程序设计
升序排,降序排。
仅介绍无符号数据升序排。
冒泡法:
相邻数互换的排序方法,类似水中气泡上浮。
排序时从前向后进行相邻两个数的比较,次序与要求的顺序不符时,就将两个数互换;顺序符合要求不互换。
假设有7个原始数据的排列顺序为:
6、4、1、2、5、7、3。
第一次冒泡的过程是:
6、4、1、2、5、7、3;原始数据的排列
4、6、1、2、5、7、3;逆序,互换
4、1、6、2、5、7、3;逆序,互换
4、1、2、6、5、7、3;逆序,互换
4、1、2、5、6、7、3;逆序,互换
4、1、2、5、6、7、3;正序,不互换
4、1、2、5、6、3、7;逆序,互换,第一次冒泡结束
如此进行,各次冒泡的结果如下:
第1次冒泡结果:
4、1、2、5、6、3、7
第2次冒泡结果:
1、2、4、5、3、6、7
第3次冒泡结果:
1、2、4、3、5、6、7
第4次冒泡结果:
1、2、3、4、5、6、7;已完成排序
第5次冒泡结果:
1、2、3、4、5、6、7
第6次冒泡结果:
1、2、3、4、5、6、7
对于n个数,理论上应进行(n-1)次冒泡,有时不到(n-1)次就已完成排序。
如何判定排序是否已完成,看各次冒泡中是否有互换发生,如果有数据互换,则排序还没完成。
在程序设计中,常使用设置互换标志的方法,该标志的状态表示在一次冒泡中是否有互换进行。
例4-10一批单字节无符号数,以R0为首地址指针,R2中为字节数,将这批数进行升序排列。
程序框图如图4-2所示。
SORT:
MOVA,R0;
MOVR1,A
MOVA,R2;字节数送入R5
MOVR5,A
CLRF0;互换标志位F0清零
DECR5;
MOVA,@R1;
LOOP:
MOVR3,A;
INCR1;
CLRC;
MOVA,@R1;比较大小
SUBBA,R3;
JNCLOOP1;
SETBF0;互换标志位F0置1
MOVA,R3;;
XCHA,@R1;两个数互换
DECR1;
XCHA,@R1;
INCR1
LOOP1:
MOVA,@R1
DJNZR5,LOOP
JBF0,SORT
RET
备注