第 4章 MCS-51汇编语言程序设计汇编语言是面向机器硬件的语言,要求程序设计者对
MCS-51单片机具有很好的,软、硬结合,的功底。
介绍 程序设计的基本知识 及如何 使用汇编语言来进行基本的程序设计。
4.1 汇编语言程序设计概述
4.1.1 机器语言、汇编语言和高级语言用于程序设计的 语言基本上分为 3种:机器语言、汇编语言和高级语言 。
1.机器语言二进制代码表示的指令、数字和符号简称为机器语言不易懂,难记忆,易出错。
2.汇编语言英文助记符表示的指令称为 符号语言 或 汇编语言将汇编语言程序转换成为二进制代码表示的机器语言程序称为 汇编程序经汇编程序,汇编(翻译),得到的机器语言程序称为 目标程序,原来的汇编语言程序称为 源程序 。
汇编语言特点,
(1)面向机器的语言,程序设计员须对 MCS-51的硬件有相当深入的了解。
(2)助记符指令和机器指令一一对应,用汇编语言编写的 程序效率高,占用 存储空间小,运行 速度快,
用汇编语言能编写出最优化的程序。
能直接管理和控制硬件设备(功能部件),它能处理中断,也能直接访问存储器及 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,MOV A,#00H ; 0→A
MOV R1,#10 ; 10→R 1
MOV R2,#00000011B ; 3→R 2
LOOP,ADD A,R2 ; ( A) +( R2) → A
DJNZ R1,LOOP; R1内容减 1不为零,则循环
NOP
HERE,SJMP HERE
基本语法规则:
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) 美元符号 $的使用用于表示该转移指令操作码所在的地址 。 例如,如下指令:
JNB F0,$
与如下指令是等价的:
HERE,JNB F0,HERE
再如:
HERE,SJMP HERE
可写为:
SJMP $
4,注释字段必须以分号,;,开头,换行书写,但必须注意也要以分号,;,开头。
汇编时,注释字段不会产生机器代码。
4.1.3 伪指令在 MCS-51 汇编语言源程序中应有向汇编程序发出的指示信息,告诉它如何完成汇编工作,这是通过使用伪指令来实现的。
也称为汇编程序控制命令。只有在汇编前的源程序中才有伪指令。经过汇编得到目标程序(机器代码)
后,伪指令已无存在的必要,所以,伪,体现在汇编时,伪指令没有相应的机器代码产生 。
常用的伪指令,
1.ORG( ORiGin)汇编起始地址命令在汇编语言源程序的开始,通常都用一条 ORG伪指令来实现规定程序的起始地址。如不用 ORG规定,则汇编得到的目标程序将从 0000H开始。 例如:
ORG 2000H
START,MOV A,#00H

规定标号 START代表地址为 2000H开始 。
在一个源程序中,可多次使用 ORG指令,来规定不同的程序段的起始地址 。 但是,地址必须由小到大排列,地址不能交叉,重叠 。 例如:
ORG 2000H

ORG 2500H

ORG 3000H

2,END(END of assembly)汇编终止命令汇编语言源程序的结束标志,用于终止源程序的汇编工作。在整个源程序中只能有一条 END命令,且位于程序的最后。
3,DB( Define Byte)定义字节命令在程序存储器的连续单元中定义字节数据。
ORG 2000H
DB 30H,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( Define Word)定义数据字命令从指定的地址开始,在程序存储器的连续单元中定义
16位的数据字 。 例如:
ORG 2000H
DW 1246H,7BH,10
汇编后:
( 2000H) =12H ;第 1个字
( 2001H) =46H
( 2002H) =00H ;第 2个字
( 2003H) =7BH
( 2004H) =00H ;第 3个字 ( 2005H) =0AH
( 2005H) =0AH
5,EQU( EQUate)赋值命令用于给标号赋值。赋值以后,其标号值在整个程序有效。例如:
TEST EQU 2000H
表示标号 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)绝对调用指令,ACALL addr11
( 2)长调用指令,LCALL addr16
3.注意设置堆栈指针和现场保护
4.最后一条指令必须是 RET指令
5.子程序可以嵌套,即子程序可以调用子程序
6.在子程序调用时,还要注意参数传递的问题二,子程序的基本结构
MAIN,┇ ; MAIN为主程序或调用程序标号

LCALL SUB ;调用子程序 SUB


SUB,PUSH PSW ;现场保护
PUSH ACC ;
子程序处理程序段
POP ACC ;现场恢复
POP PSW ;
RET ;最后一条指令必须为 RET
例 4-3 单字节有符号数的加减法子程序(自己阅读)
本例中参数传递是通过累加器 A完成的,主程序将被转换的数送到 A中,子程序将 A中的有符号数求补后存于 A中,主程序再将结果放回原来的单元。
例 4-4 4位 BCD码的减法程序(自己阅读)
主程序通过地址寄存器 R0和 R1将参加运算的 BCD码的地址传递给子程序,子程序则通过累加器将差传递给主程序。
4.3.3 查表程序设计数据补偿、修正、计算、转换等各种功能,具有程序简单、执行速度快等优点。
查表就是根据自变量 x,在表格中寻找 y,使 y=f(x)。
执行查表指令时,发出读程序存储器选通脉冲 /PSEN。
在 MCS-51的指令系统中,给用户提供了 两条极为有用的查表指令:
⑴ MOVC A,@A+DPTR
⑵ MOVC A,@A+PC
指令,MOVC A,@A+DPTR”完成把 A中的内容作为一个无符号数与 DPTR中的内容相加,所得结果为某一程序存储单元的地址,然后把该地址单元中的内容送到累加器 A中。
指令,MOVC A,@A+PC”以 PC作为基址寄存器,PC的内容和 A的内容作为无符号数,相加后所得的数作为某一程序存储器单元的地址,根据地址取出程序存储器相应单元中的内容送到累加器 A中。
指令执行完,PC的内容不发生变化,仍指向查表指令的下一条指令。优点在于预处理较少且不影响其它特殊功能寄存器的值,所以不必保护其它特殊功能寄存器的原先值。缺点在于该表格只能存放在这条指令的地址 X3X2X1X0以下的 00~ FFH之中。表格所在的程序空间受到了限制。
例 4-5 子程序的功能为:根据累加器 A中的数 x( 0~
9之间)查 x的平方表 y,根据 x的值查出相应的平方
y。 x和 y均为单字节数。
地 址 子程序
Y3Y2Y1Y0 ADD A,#01H
Y3Y2Y1Y0+2 MOVC A,@A+PC
Y3Y2Y1Y0+3 RET
Y3Y2Y1Y0+4 DB 00H,01H,04H,09H,10H
DB 19H,24H,31H,40H,51H
第 1条指令 ADD A,#01H 的作用是加上偏移量,可以根据 A的内容查出 X对应的平方 。
MOVC A,@+DPTR 这条指令的应用范围较为广泛,一般情况下,大多使用该指令,使用该指令时不必计算偏移量,使用该指令的优点是表格可以设在 64K程序存储器空间内的任何地方,而不像 MOVC A,@A+PC
那样只设在 PC下面的 256个单元中,使用较方便。
上面的程序可改成如下形式:
PUSH DPH ;保存 DPH
PUSH DPL ;保存 DPL
MOV DPTR,#TAB1
MOVC A,@A+DPTR
POP DPL ;恢复 DPL
POP DPH ;恢复 DPH
RET
TAB1,DB 00H,01H,04H,09H,10H
DB 19H,24H,31H,40H,51H
例 4-6 在一个以 MCS-51为核心的温度控制器中,温度传感器输出的电压与温度为非线性关系,传感器输出的电压已由 A/D转换为 10位二进制数 。 根据测得的不同温度下的电压值数据构成一个表,表中放温度值 y,x为电压值数据 。 设测得的电压值 x放入 R2R3中,
根据电压值 x,查找对应的温度值 y,仍放入 R2R3中 。
本例的 x和 y均为双字节无符号数 。 程序如下:
LTB2,MOV DPTR,#TAB2
MOV A,R3
CLR C
RLC A
MOV R3,A
XCH A,R2
RLC A
XCH R2,A
ADD A,DPL ; ( R2R3) +( DPTR)
→ ( DPTR)
MOV DPL,A
MOV A,DPH
ADDC A,R2
MOV DPH,A
CLR A
MOVC A,@A+DPTR ;查第一字节
MOV R2,A ;第一字节存入 R2中
CLR A
INC DPTR
MOVC A,@A+DPTR ;查第二字节
MOV R3,A ;第二字节存入 R3中
RET
TAB2:DW…… ;温度值表例 4-7 设有一个巡回检测报警装置,需对 16路输入进行检测,每路有一最大允许值,为双字节数 。 运行时,需根据测量的路数,找出每路的最大允许值 。
看输入值是否大于最大允许值,如大于就报警 。 根据上述要求,编一个查表程序 。
取路数为 x(0x15),y为最大允许值,放在表格中 。
设进入查表程序前,路数 x已放于 R2中,查表后最大值 y放于 R3R4中 。 本例中的 x为单字节数,y为双字节数 。 查表程序如下:
TB3,MOV A,R2
ADD A,R2 ; (R2)*2→(A)
MOV R3,A ;保存指针
ADD A,#6 ;加偏移量
MOVC A,@A+PC ;查第一字节
XCH A,R3
ADD A,#3
MOVC A,@A+PC ;查第二字节
MOV R4,A
RET
TAB3,DW 1520,3721,42645,7580 ;最大值;表
DW 3483,32657,883,9943
DW 10000,40511,6758,8931
DW 4468,5871,13284,27808
表格长度不能超过 256个字节,且表格只能存放于 MOVC
A,@A+PC指令以下的 256个单元中。
4.3.4 关键字查找程序设计顺序检索和对分检索一、顺序检索从第 1项开始逐项顺序查找,判断所取数据是否与关键字相等。
例 4-8 从 50个字节的无序表中查找一个关键字 ××,H。
ORG 1000H
MOV 30H,#×× H ;关键字 ×× H送 30H单元
MOV R1,#50 ;查找次数送 R1
MOV A,#14 ;修正值送 A
MOV DPTR,#TAB4 ;表首地址送 DPTR
LOOP,PUSH ACC
MOVC A,@ A+PC ;查表结果送 A
CJNE A,40H,LOOP1;( 40H)不等于关键字则转 LOOP1
MOV R2,DPH ;已查到关键字,把该字;的地址送 R2,R3
MOV R3,DPL ;
DONE,RET
LOOP1,POP ACC ;修正值弹出
INC A ; A+1→A
INC DPTR ;修改数据指针 DPTR
DJNZ R1,LOOP ; R1≠0,未查完,继续查找
MOV R2,#00H ; R1=0,清,0” R2 和 `R3
MOV R3,#00H ;表中 50个数已查完
AJMPDONE ;从子程序返回
TAB4,DB …,…,… ; 50个无序数据表二、对分检索前提,检索的数据表已经排好序,如何进行数据的排序,将在本节稍后介绍。
方法,取数据表中间位置的数与关键字进行比较,如相等,则查找到;如果所取的数大于关键字,则下次对分检索的范围是从数据区起点到本次取数。如果取数小于关键字,则下次对分检索的范围是从本次取数数据区起点到数据区终点。依此类推,逐渐缩小检索范围,减少次数,大大提高了查找速度。
4.3.5 数据极值查找程序设计在指定的数据区中找出最大值(或最小值)。
进行数值大小的比较,从这批数据中找出最大值(或最小值)并存于某一单元中。
例 4-9 片内 RAM中存放一批数据,查找出最大值并存放于首地址中。设 R0中存首地址,R2中存放字节数,
程序框图如图 4-1所示。
程序如下:
MOV R2,n ; n为要比较的数据字节数
MOV A,R0 ;存首地址指针
MOV R1,A
DEC R2 ;
MOV A,@R1
LOOP,MOV R3,A
DEC R1
CLR C
SUBB A,@R1 ;两个数比较
JNC LOOP1 ; C=0,A中的数大,跳 LOOP1
MOV A,@R1 ; C=1,则大数送 A
SJMP LOOP2
LOOP1:MOV A,R3
LOOP2:DJNZ R2,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,MOV A,R0 ;
MOV R1,A
MOV A,R2 ;字节数送入 R5
MOV R5,A
CLR F0 ;互换标志位 F0清零
DEC R5 ;
MOV A,@R1 ;
LOOP,MOV R3,A ;
INC R1 ;
CLR C ;
MOV A,@R1 ;比较大小
SUBB A,R3 ;
JNC LOOP1 ;
SETB F0 ;互换标志位 F0置 1
MOV A,R3; ;
XCH A,@R1 ;两个数互换
DEC R1 ;
XCH A,@R1 ;
INC R1
LOOP1,MOV A,@R1
DJNZ R5,LOOP
JB F0,SORT
RET
4.3.7 分支转移程序设计特点是程序中含有转移指令,转移指令有分为 无条件转移 和 有条件转移,因此分支程序也可分为无条件分支转移程序和有条件分支转移程序。有条件分支转移程序按结构类型来分,又分为 单分支转移 结构和 多分支转移 结构 。
一、分支转移结构
1.单分支转移结构仅有两个出口,两者选一。
例 4-11 求单字节有符号数的二进制补码参考程序,
CMPT,JNB Acc.7,RETURN; ( A) >0,不需转换
MOV C,Acc.7 ;符号位保存
CPL A ; ( A) 求反,加 1
ADD A,#1 ;
MOV Acc.7,C ;符号位存 A的最高位
RETURN,RET
此外,单分支选择结构还有如图 4-4,图 4-5等所示的几种形式:
2.多分支转移结构程序的判别部分有两个以上的出口流向。
常见的两种形式。如图 4-6和图 4-7。
指令系统提供了非常有用的两种多分支选择指令,
间接转移指令,JMP @A+DPTR;
比较转移指令,CJNE A,direct,rel;
CJNE A,#data,rel;
CJNE Rn,#data,rel;
CJNE @Ri,#data,rel;
最简单的分支转移程序的设计,一般常采用逐次比较法,就是把所有不同的情况一个一个的进行比较,
发现符合就转向对应的处理程序。这种方法的主要缺点是程序太长,有 n种可能的情况,就需有 n个判断和转移。
例 4-12 求符号函数的值。
程序框图如图 4-6所示。
程序 略。
在实际的应用中,经常遇到的图 4-7结构形式的分支转移程序的设计 。
需根据某一单元的内容是 0,1,……,n,来分别转向处理程序 0,处理程序 1,…… 处理程序 n。一个典型的例子就是当单片机系统中的键盘按下时,
就会得到一个键值,根据不同的键值,跳向不同的键处理程序入口。对于这种情况,可用直接转移指令( LJMP或 AJMP指令)组成一个转移表,然后把该单元的内容读入累加器 A,转移表首地址放入 DPTR中,再利用间接转移指令实现分支转移。
例 4-12 根据寄存器 R2的内容,转向各个处理程序
PRGX( X=0~n) 。
( R2) =0,转 PRG0
( R2) =1,转 PRG1

( R2) =n,转 PRGn
4.3.8 循环程序设计特点是程序中含有可以反复执行的程序段,该程序段通常称为循环体。例如求 100个数的累加和,则没有必要连续安排 100条加法指令,可以只用一条加法指令并使其循环执行 100次。
( 1)可大大缩短程序长度( 2)使程序所占的内存单元数量少( 3)使程序结构紧凑和可读性变好。
一、循环程序的结构循环结构程序主要由以下四部分组成。
1.循环初始化循环初始化程序段用于完成循环前的的准备工作。例如,循环控制计数初值的设置、地址指针的起始地址的设置、为变量预置初值等。
2,循环处理循环程序结构的核心部分,完成实际的处理工作,是需反复循环执行的部分,故又称循环体 。 这部分程序的内容,取决于实际处理问题的本身 。
3,循环控制在重复执行循环体的过程中,不断修改循环控制变量,
直到符合结束条件,就结束循环程序的执行 。 循环结束控制方法分为循环计数控制法和条件控制法
4,循环结束这部分是对循环程序执行的结果进行分析、处理和存放。
二、循环结构的控制图 4-8是计数循环控制结构,图 4-9是条件控制结构。
1.计数循环结构计数循环控制结构是依据计数器的值来决定循环次数,一般为减,1” 计数器,计数器减到,0”时,
结束循环。计数器的初值是在初始化时设定。
MCS-51的指令系统提供了功能极强的循环控制指令:
DJNZ Rn,rel;工作寄存器作控制计数器
DJNZ direct,rel;以直接寻址单元作控制计数器。
例如,计算 n个数据的和。
程序框图见图 4-10 。
计数控制方法只有在循环次数已知的情况下才适用。对循环次数未知的问题,不能用循环次数来控制。往往需要根据某种条件来判断是否应该终止循环。
2,条件控制结构例 4-15 设有一串字符,依次存放在内部 RAM从 30H单元开始的连续单元中,该字符串以 0AH为结束标志,
编写测试字符串长度的程序 。
本例采用逐个字符依次与,0AH”比较的方法 。 为此设置一个长度计数器和一个字符串指针 。 长度计数器用来累计字符串的长度,字符串指针用于指定字符 。
如果指定字符与,0AH”不相等,则长度计数器和字符串指针都加 1,以便继续往下比较;如果比较相等,则表示该字符为,0AH”,字符串结束,长度计数器的值就是字符串的长度 。 程序略 。
最常见的多重循环是由 DJNZ指令构成的软件延时程序,
它是常用的程序之一 。
例 4-16 50ms延时程序 。
延时程序与 MCS-51指令执行时间有很大的关系 。
在使用 12MHz晶振时,一个机器周期为 1μs,执行一条 DJNZ指令的时间为 2μs。 这时,可用双重循环方法写出下面如下的延时 50ms的程序:
DEL,MOV R7,#200
DEL1,MOV R6,#125
DEL2,DJNZ R6,DEL2 ; 125*2=250μs
DJNZ R7,DEL1 ; 0.25ms*200=50ms
RET
以上延时程序不太精确,它没有考虑到除,DJNZ
R6,DEL2,指令外的其它指令的执行时间,如把其它指令的执行时间计算在内,它的延时时间为:
( 250+1+2) *200+1=50.301ms
如果要求比较精确的延时,可按如下修改:
DEL,MOV R7,#200
DEL1,MOV R6,#123
NOP
DEL2,DJNZ R6,DEL2 ; 2*123+2=248μs
DJNZ R7,DEL1 ; (248+2)*200+1=50.001ms
RET
实际延迟时间为 50.001ms,注意 。 软件延时程序,不允许有中断,否则将严重影响定时的准确性 。
4.3.9 码制转换程序设计在单片机应用程序的设计中,经常涉及到各种码制的转换问题。在单片机系统内部进行数据计算和存储时,经常采用二进制码,具有运算方便、存储量小的特点。在输入 /输出中,按照人的习惯均采用代表十进制数的 BCD码(用 4位二进制数表示的十进制数)表示。此外,打印机要打印某数字字符,则需要将该数字的二进制码转换为该字符的 ASCII码,
才能送到打印机去打印。
一、二进制码到 BCD码的转换
BCD码有两种形式:一种是 1个字节放 1位 BCD码,它适用于显示或输出,一种是 压缩的 BCD码,即 1个字节放两位 BCD码,可以节省存储单元。
二,BCD码到二进制码的转换三、二进制码与 ASCII码之间的转换