1、汇编语言源程序的格式 1.内部数据传送指令 汇编语言是面向机器的,因此,语言格式因机器不同而异。对MCS-51系统来说,汇编语言中每条语句的格式包括下列4项内容: 标号: 操作码 操作数; 注释 汇编语句中,标号和操作码要用冒号“:”隔开;操作码和操作数之间的分隔符是空格,多个操作数之间用“,”分隔;操作数与注释之间用“;”分隔;操作码是必选项,其余都是可选项,即任何语句都必须包含操作码,其他部分因语句不同而异。 1.ORG伪指令 ORG伪指令称为起始汇编伪指令,一般用于汇编语言源程序或某数据块的开头,格式为: [标号]:ORG 16位的地址或标号 2.END伪指令 END伪指令称为汇编结束伪指令,经常用在汇编语言源程序的末尾,用来指示源程序结束汇编的位置。即表明程序的结束。一般格式为: [标号]:END 3.EQU伪指令 EQU伪指令称为赋值伪指令,用于给左边的“字符名”赋值。此伪指令的格式为: 字符名 EQU 数据或汇编符号 4.DATA伪指令 DATA伪指令称为数据地址赋值伪指令,它用来给左边的“字符名”赋值。其一般格式为: 字符名 DATA 数据或表达式 5.BIT伪指令 BIT伪指令称为位地址符号伪指令,用来给符号形式的位地址赋值,此伪指令的格式为: 字符名 BIT 位地址 6.DB伪指令 DB伪指令称为定义字节伪指令,它的功能是从指定单元开始定义(存储)若干个字节的数据或字符,字符若用引号括起来则表示ASCII码。其一般格式为: 标号:DB 字节常数或字符 7.DW伪指令 DW伪指令称为定义字伪指令,其功能为在程序存储器中从指定单元开始,定义若干个字,一个字相当于两个字节。此伪指令的一般格式为: 标号: DW 字常数或字表 8.DS伪指令 DS伪指令称为定义存储空间伪指令,格式为: 标号: DS 表达式2、MCS―51单片机汇编语言程序设计举例 1. 简单程序设计 例:将一个字节内的两位压缩BCD码拆开并转换成相应的ASCH码,存入两个RAM单元。 解:设两位压缩BCD码已放在内部RAM的20H单元,转换后的ASCII码放在21H和22H单元。根据ASCII码表,字符0~9对应的ASCII码为30H~39H,之间仅相差30H。因此,转换时,只需把20H单元中两位压缩BCD码拆开后,将BCD的高四位置成“0011”即可。相应程序如下: ORG 1000H MOV R0, #20H MOV A, @R0 ;两位BCD码送A PUSH ACC ANL A, #0FH ;取低位BCD码 ORL A, #30H ;完成低位转换 INC R0 MOV @R0, A ;低位BCD码的转换结果存入21H中 POP ACC ANL A ,#0F0H ;取高位BCD码 SWAP A ORL A, #30H ;完成高位转换 INC R0 MOV @R0, A ;存数 SJMP $ ;结束 END 2. 分支程序设计 例:设变量X存放于R2,函数值Y存放在R3。试按照下式的要求给Y赋值:
解:这是一个三分支的条件转移程序,可采用CJNE和JC或JNC指令进行判断。 ORG 0500H MOV A, R2 ;自变量→(A) CJNE A,#10,L1 ;(A)与10比较 L1:JC L2 ;若X<10,则转L2 ADD A, #01H MOV R3, A ; 设X>20,Y=1 CJNE A,#21,L3 L3:JNC L4 ;X>20,则转L4 MOV R3,#0 ;20≥X≥10,Y=0 SJMP L4 L2:MOV R3,#0FFH L4:SJMP $ END 3. 循环程序设计 循环程序一般由以下几部分组成: 1)循环初始化部分 2)循环体部分 3)循环结束部分 例:在内部RAM的20H~2FH连续16个单元中存放单字节无符号数。求16个无符号数之和。 解 这是重复相加问题。16个单字节数的和最大不会超过两个字节,设和存放在31H,30H中。用R0作加数指针,R7作循环次数计数器。程序流程如图3-11所示。 ORG 1000H MOV R7,#0FH MOV R0,#21H MOV 31H,#00H MOV A,20H LOOP1: ADD A,@R0 MOV 30H, A JNC LOOP2 INC 31H LOOP2: INC R0 DJNZ R7, LOOP1 SJMP $ END 4.查表程序设计 查表程序是根据查表算法设计的。它有两条专门的查表指令: 例:设计一个将16进制数转换成ASCII码的子程序。设16进制数存放在R0中的低4位,要求将转换后的ASCII码送回R0中。 解:给出二种方案。 ①计算求解。由ASCII码字符表可知0~9的ASCII码为30H~39H,A~F的ASCII码为41H~46H。因此,计算求解的思路是:若(R0)≤9,则R0内容只需加30H;若(R0)>9,则R0需加37H。相应程序为: ORG 1000H MOV A,R0 ;取转换值到A ANL A,#0FH ;屏蔽高4位 CJNE A,#10,NEXTl NEXTl:JNC NEXT2 ;若A>9,则转NEXT2 ADD A,#30H ;若A<10,则A (A)+30H SJMP DONE NEXT2:ADD A,#37H ;A (A)+37H DONE: MOV R0,A ;存结果 SJMP $ END ②查表求解。求解时,两条查表指令任选其一。现以“MOVC A,@A+PC”指令为例,给出相应程序: 地址 机器码 ORG 1000H 1000 E8 MOV A,R0 ;取转换值 1001 54 0F ANL A,#0FH ;屏蔽高4位 1003 24 03 ADD A,#03H ;计算偏移量 1005 83 MOVC A,@A+PC ;查表 1006 F8 MOV R0,A ;存结果 1007 80 FE SIMP $ 1008 30 31 32 33 ASCTAB: DB 30H,31H,32H,33H 100C 34 35 36 37 DB 34H,35H,36H,37H 1010 38 39 41 42 DB 38H,39H,41H,42H 1014 43 44 45 46 DB 43H,44H,45H,46H END 5. 子程序设计 子程序在结构上应具有通用性和独立性,在编写子程序时应注意: ①程序第一条指令的地址称为入口地址,该指令前必须有标号,最好以子程序任务名作为标号,例如显示程序常以DIR作为标号; ②调用子程序指令设在主程序中,在子程序的末尾一定要有返回指令。一般说来,子程序调用指令和子程序返回指令要成对使用,子程序应只有一个出口; ③子程序调用和返回指令能自动保护和恢复断点地址,但对需要保护的寄存器和内存单元的内容,必须在子程序开始和末尾(RET指令前)安排保护和恢复它们的指令; ④调用子程序时,要了解子程序的“入口信息”和“出口信息”,即进入子程序前应给哪些变量赋值,子程序返回时结果存在何处,以便主程序应用这些结果。这就是所谓的参数传递。一般称传入子程序的参数为入口参数,由子程序返回的参数为出口参数。 例:用程序实现C=a2+b2。设a、b均小于10。a存在21H单元,b存在22H单元,结果C存在20H单元。 解:因本题中两次用到求平方的运算,故此把求平方运算编成子程序。依题意编写主程序和子程序如下: ORG 1000H MAIN: MOV SP,#60H ;设堆栈指针 MOV A,21H ;取a值 LCALL SQR ;求a2 MOV 20H,A ;a2值送入20H单元 MOV A,22H ;取b值 LCALL SQR ;求b2 ADD A,20H ;求a2+b2 MOV 20H,A ;结果存入20H单元 SJMP $ ORG 2000H SQR: MOV B,A ;求平方子程序 MUL AB RET |