《《汇编语言程序》PPT课件.ppt》由会员分享,可在线阅读,更多相关《《汇编语言程序》PPT课件.ppt(122页珍藏版)》请在三一办公上搜索。
1、1,第四章 汇编语言程序格式,教学要点:,理解汇编语言的基本格式掌握汇编语言中的数据类型、运算符、表达式 的表示方法和作用掌握伪指令的格式和作用理解汇编语言源程序的结构了解汇编语言的上机过程,2,汇编语言是面向机器的低级程序设计语言。是一种符号语言,它用指令的助记符,符号地址,标号等各种符号来书写程序。汇编语言有自汇编程序、交叉汇编程序。汇编语言直观易读,直接控制硬件的最底层。,3,4.1 汇编语言语句种类及其格式,不同的汇编程序有不同的汇编语言编程规定。目前支持Intel8086/8088系列微机,常用的汇编程序有ASM、MASM、TASM、OPTASM等。,用汇编语言编写程序,首先要了解汇
2、编语言的语法规则和语句格式,“语句”是源程序的基本单位。本章主要介绍汇编语言程序设计中的一些基本书写格式与语法规则。,汇编语言的语句可以分为指令语句、伪指令语句和宏指令语句(宏调用语句)。,4,指令语句是可执行的,在汇编时都要产生一个可供CPU执行的机器目标代码。例:INC AX,一、指令语句,一条指令语句最多可以包含4个字段,5,名字是可选字段,它后面必须有“:”。名字是一条指令的符号地址,代表了该指令的第一个字节存放地址。,1.名字,名字一般放在一个程序段或子程序的入口处,控制程序的执行转到该程序位置。,例:ADDR1:MOV AL,100,在转移指令或子程序调用指令中,可直接引用这个名字
3、。,6,在汇编语言中,允许使用如下字符做为名字:字母:a-z,A-Z数字:0-9字符:?;._$:,名字是一个字符串序列,最多包含31个字符。名字可以作为标号或者是变量。,7,该字段是一条指令的必选项,它表示这条语句要求CPU完成什么具体操作,如MOV、ADD、SHL等。为语句的核心成分,表示了该语句的操作类型。,2.操作码,有些指令还可以在指令助记符的前面加上前缀,实现一定的附加操作。如串操作指令前所加的重复前缀REP等。,8,操作数有一个或者多个表达式组成。一条指令可以有一个操作数、两个操作数或者无操作数。,3.操作数字段,如ADD、MOV指令需要两个操作数,INC、NOT指令只需一个操作
4、数,而CLC指令不需要操作数。,操作数可以是常数,寄存器,标号,或者表达式组成。,9,4.注释字段,例:LABEL1:ADD AX,BX;功能为AX=(AX)+(BX);后面的程序段将完成一次对存储器的访问,注释字段为可选项,该字段以分号“;”开始。,注释字段可以是一条指令的后面部分,也可以是整个语句行。,它的作用是为阅读程序的人加上一些说明性内容,注释字段不会产生机器目标代码,它不会影响程序和指令的功能。,10,伪指令是不可执行语句,本身并不产生对应的机器目标代码。它仅仅是告诉汇编程序对其后面的指令语句和伪指令语句的操作数应该如何处理。,二、伪指令语句,伪指令语句又叫命令语句。,一条伪指令语
5、句可以包含四个字段。如下所示:,例:abc db?,11,符号名:符号名是伪指令语句的一个可选项。伪指令符:伪指令符指定汇编程序要完成的具体操作,如数据定义伪指令DB、DW、DD,段定义伪指令SEGMENT,假定伪指令ASSUME等。操作数:伪指令后面的操作数可以是常数、字符串、变量、表达式等,其个数由具体的伪指令决定,各个操作数之间必须以“逗号”分隔。注释:伪指令的注释必须以“;”开始,其作用同指令语句中的注释部分。,12,三、标识符,指令语句中的名字和伪指令语句中符号名统称为标识符。标识符是由若干个字符构成的。,标识符构成规则:,1.字符的个数为131个;,2.第一个字符必须是字母、问号、
6、或下划线“_”这4种字符之一;,13,保留字主要有CPU中各寄存器名(如AX、CS等),指令助记符(如MOV、ADD),伪指令符(如SEGMENT、DB)、表达式中的运算符(如GE、EQ)以及属性操作符(如PTR、OFFSET等)。,3.不能使用属于系统专用的保留字。,14,四、汇编语言的程序格式,使用汇编语言编写程序时必须遵循的语法规则。完整的汇编语言源程序由段组成:若干个代码段、数据段、附加段或堆栈段;独立运行的程序必须包含一个代码段,并指出程序的启动语句;所有的执行语句必须位于某一个代码段内,说明性语句可根据需要位于任一段内;堆栈段不是必须的,但最好设置堆栈段。,15,编写汇编语言源程序
7、通常应遵循所谓“标准格式”,现在有两种表示形式,一种符合MASM 5.0及更新版本的语法规则,称为“简化段定义格式”;另一种符合早期版本的语法规则,称为“完整段定义格式”(5.0及以后的版本也支持)。,16,简化段定义标准格式:.model small.stack.data.code.startup.exit 0end,;子程序代码,;程序代码,;数据定义,程序开始伪指令,程序结束伪指令,存储模式伪指令,汇编结束伪指令,17,简化段定义格式,.model small.stack.data string db hello.code.startup;说明程序的起点,设置DS、SS mov dx,o
8、ffset,string mov ah,0.h9 int 21h.exit 0;程序结束,形成返回DOS的指令 end,MASM5.0 5.1不支持,18,data segment;定义数据段 data ends;-extra segment;定义附加段 extra ends;-stack segment;定义堆栈段 stack ends;-code segment;定义代码段 assume cs:code,ds:data,es:extrastart:mov ax,data mov ds,ax;段地址 段寄存器 code ends end start,完整的段定义格式,19,举例:编写一个汇编
9、语言程序,实现下列公式计算。假设X=4,Y=5(XY)8X 2,Z=,20,设计思路一:(1)设公式中出现的三个变量X、Y、Z是8位带符号数;(2)用算术运算指令实现加减乘除运算;(3)指令顺序按照运算顺序书写。,21,程序段:MOV AL,X;ALXADD AL,Y;ALX+Y 加法MOV BL,8;BL8IMUL BL;AXAL8 乘法MOV BL,X;BLXMOV BH,0;BH0SUB AX,BX;AXAXX 减法MOV BL,2;BL2IDIV BL;AX2 除法,商在AL,余数在AH中MOV Z,AL;Z商MOV Z1,AH;Z1余数,22,设计思路二:(1)公式中出现的三个变量X
10、、Y、Z是带符号数,在数据段中定义;(2)用算术运算指令实现加减运算;(3)将操作数左移3位二进制位数代替乘以8运算,操作数右移1位相当于除以2;(4)指令在代码段中,指令顺序按照运算顺序书写。,23,DATA SEGMENT;数据段定义伪指令X DW 4;定义X为字单元,值为4Y DW 5Z DW?;定义Z为空单元DATA ENDS;数据段结束CODE SEGMENT;代码段定义伪指令 ASSUME CS:CODE,DS:DATA;指定段寄存器与对应段名,完整的程序如下:,24,START:MOV AX,DATA MOV DS,AX;将数据段段地址送入DSMOV BX,X MOV AX,YA
11、DD AX,BX;加法MOV CL,3SAL AX,CL;算术左移3次,相当于乘以8SUB AX,X;减法SAR AX,1;算术右移1次,相当于除以2MOV Z,AXMOV AH,4CH;此两句为结束程序,返回DOSINT 21H CODE ENDS;代码段结束 END START;整个程序结束伪指令,25,Data segment string db helloData endsStack segment st1 dw 512 dup(?)Stack endsCode segment assume cs:code,ds:data,ss:stackStart:mov ax,data mov d
12、s,ax,输出一个字符串,26,mov dx,offset string mov ah,09h int 21h mov ah,4ch int 21h Code ends End start,输出一个字符串,27,DATA SEGMENTBUF1 DB 34HBUF2 DB 2AHSUM DB?DATA ENDSCODE SEGMENT ASSUME CS:CODE,DS:DATASTART:MOV AX,DATA MOV DS,AX MOV AL,BUF1 ADD AL,BUF2 MOV SUM,AL,MOV AH,4CHINT 21HCODE ENDS END START,两个字节的数据相加
13、,28,通常一个汇编程序有3个逻辑段组成:数据段、堆栈段、代码段。,(1)数据段用来在内存中建立一个适当容量的工作区,存放常数、变量等程序需要对其进行操作的数据。,(2)堆栈段建立的区域,以便在中断、子程序调用时使用。堆栈区域不能太小,否则出现溢出。,(3)代码段包括许多以符号表示的指令,其内容就是程序要执行的指令。,29,作为汇编程序主模块还必须有下面部分:,使用ASSUME伪指令告诉汇编程序,某一段地址应该放在哪个段寄存器。,需要返回DOS语句。MOV AH,4CH INT 21H,DOS的装入程序在执行时,需要对程序执行时装入正确地址,所以初始化。MOV AX,DATA MOV DS,A
14、X,30,在说明数据时不仅要指定其数值,还需说明它的属性,如是字节数据还是字数据。,4.2 汇编语言数据,数据是指令和伪指令语句中操作数的基本组成部分。一个数据由数值和属性两部分构成。,在汇编语言中常用的数据形式有:常数、变量和标号。,31,常数在汇编期间其值已完全确定,并且在程序运行过程中,其值不会发生变化。,一、常数,常数有以下几种形式:,1.二进制数:以字母B结尾,如01001001B,2.八进制数:以字母O或Q结尾,如631Q、254O,3.十进制数:以字母D结尾,或者没有结尾字母。如2007D、2007。,32,4.十六进制数:以字母H结尾,如3FEH,如果常数的第一个数字为字母,为
15、了与标识符加以区别,必须在其前面冠以数字“0”。,整数部分 小数部分E 指数部分,汇编程序在汇编源程序时,可以把实数转换为4字节、8字节或10字节的二进制数形式存放。,例:2.134 E+10,5.实数。一般格式为:,33,6字符串常数:用引号(单引号或双引号)括起来的一个或多个字符,这些字符以它的ASCII码值存储在内存。,例如:B在内存中为42H,ABC为41H 42H 43H。在内存中的存储如图所示。,34,常数在程序中可以用在以下几种情况:,(1)作指令语句的源操作数,(2)在指令语句的直接寻址方式、变址(基址)寻址方式或基址变址寻址方式中作位移量。,MOV AX,0B2F0HADD
16、AH,64H,MOV BX,32H SI 寄存器相对MOV 0ABH BX,CX 寄存器相对ADC DX,1234H BPDI 相对基址变址,35,(3)在数据定义伪指令中使用,定义数据值,DATA SEGMENT DA1 DB 10H DA2 DW 3210HDATA ENDS,36,变量用来表示存放数据的存储单元,这些数据在程序运行期间可以被改变。,二、变量,程序中以变量名的形式来访问变量,因此,可以认为变量名就是存放数据的存储单元地址。,37,定义变量就是给变量在内存中分配一定的存储单元。也就是给这个存储单元赋与一个符号名,即变量名,同时还要将这些存储单元预置初置。,1.变量的定义与预置
17、,定义变量使用数据定义伪指令 DB、DW、DD、DQ和DT等。,38,变量定义的一般格式:,其中表达式1、表达式2是给存储单元赋的初值。,例如:,DATA SEGMENT DATA1 DB 12H DATA2 DB 20H,30H DATA3 DW 5678HDATA ENDS,39,DATA_BYTE DB 10,4,10H,?DATA_WORD DW 100,100H,-5,?,DATA_BYTE,DATA_WORD,例:,40,ARRAY DB HELLO DB AB DW AB,ARRAY,41,当变量被定义后,就具有了以下三个属性:,(1)段属性(SEG),该属性定义了标号和变量的段
18、起始地址,必须存放在一个段寄存器中。标号的段是它所出现的对应代码段CS,变量的段有DS或者ES指示。,例如:上面例子中的变量DATA1、DATA2和DATA3三个变量都存放在DATA逻辑段中。,42,它表示变量所在位置与段起始点之间的字节数。该数是一个16位无符号数。,(2)偏移量属性(OFFSET),段属性和偏移量属性就构造了变量的逻辑地址,如上述例子中,变量DATA1的偏移量为0,DATA2为1,DATA3为3。,43,它表示变量占用存储单元的字节数。其中DB伪指令定义的变量为字节,DW定义的变量为字,DD定义的为双字(4字节),DQ定义的为4个字,DT定义的为5个字。,(3)类型属性(T
19、YPE),44,在变量的定义语句中,给变量赋初值的表达式可以使用下面4种形式:,例如:DATA1 DB 32,30HDATA1的内容为32(20H),DATA1+1单元内容为30H。,1)数值表达式,45,例如:DA-BYTE DB?,?,?表示让汇编程序分配三个字节存储单元。这些存储单元的内容的值为任意值。,不带引号的问号“?”表示可以预置任意内容。,2)?表达式,对于DB伪指令,字符串为用引号括起来的不超过255个字符。给每一个字符分配一个字节单元。字符串按从左到右,将字符的ASCII编码值以地址递增的排列顺序依次存放。,3)字符串表达式,46,对于DW伪指令可以给两个字符组成的字符串分配
20、两个字节存储单元。,例如:STRING2 DW AB,CD,EF,例如:STRING1 DB ABCDEF,注意:两个字符的存放顺序是前一个字符放在高地址,后一字符放低地址单元。,47,例如:STRING3 DD AB,CD,注意:DW和DD伪指令不能用两个以上字符构成的字符串赋初值,否则将出错。,对于DD伪指令,只能给两个字符组成的字符串分配4个字节单元。,两个字符存放在较低地址的两个字节单元中。存放顺序与DW伪指令相同,而较高的两个字节单元都存放0。,48,4)DUP表达式,DUP称为重复数据操作符。,使用DUP表达式的一般格式为:,其中:表达式1是重复的次数,表达式2是重复的内容。,例如
21、:DATA_A DB 10H DUP(?)DATA_B DB 20H DUP(AB),分配16个字节单元,分配20H*2=40H个 字节,其内容为重复字符串AB。,49,DUP还可以嵌套使用,即表达式2又可以是一个带DUP的表达式。,例如:DATA_C DB 10H DUP(4 DUP(2),7)重复10H个数字序列“2,2,2,2,7”,共占用10H*5=50H个字节。,VAR DB 100 DUP(?)DB 2 DUP(0,2 DUP(1,2),3),数字序列“0,1,2,1,2,3,0,1,2,1,2,3”,50,2.变量的使用,(1)在指令语句中引用,例如:DA1 DB 0FEH DA
22、2 DW 52ACH MOV AL,DA1;将0FEH传送到AL中 MOV BX,DA2;将52ACH传送到BX中,在指令语句中直接引用变量名就是对其存储单元的内容进行存取。,51,当变量出现在变址(基址)寻址或基址变址寻址的操作数中时表示取用该变量的偏移量。,例如:DA3 DB 10H DUP(?)DA4 DW 10H DUP(1)MOV DA3SI,ALADD DX,DA4BXDI,将AL的内容送入从DA3 开始再偏移(SI)的存储单元中,将从DA4开始再偏移(BX)+(DI)的字存储单元的内容与DX的内容相加,结果送回DX中。,52,(2)在伪指令语句中引用,NUM DB 75HARRA
23、Y DW 20H DUP(0)ADR1 DW NUMADR2 DD NUMADR3 DW ARRAY2,它表示取变量地址的偏移量,取变量段基值和偏移量。前两个字节存偏移量,后两个字节存段基值,53,后面三条伪指令的操作数中都包含了前面定义的两个变量。,设上述语句所在段的段基值为0915H,NUM的偏移量为0004H,则存储单元的分配情况如图所示。,54,标号:写一条指令的前面,它就是该指令在内存的存放地址的符号表示,也就是指令地址的别名。是可执行语句的符号地址,一个标号对应一条指令的地址。,三、标号,标号主要用在程序中需要改变程序的执行顺序时,用来标记转移的目的地,即作转移指令的操作数。,55
24、,例如:MOV CX,100 LAB:MOV AX,BX LOOP LAB JNE NEXT;不为零转移 NEXT:,56,标号作为JMP 和CALL指令的目的操作数,与变量类似,具有段属性、偏移属性、类型。但是只有下面两点差别:标号的段基址值存在代码段寄存器(CS)中。标号的类型为NEAR和FAR,前者用于段内使用,后者用于段间使用。,57,它表示该标号可以被段内还是段间的指令调用。,每个标号具有三属性,(1)段属性(SEG),它表示该标号所代表的地址在哪个逻辑段中,即段基值。,(2)偏移量属性(OFFSET),它表示该标号所代表的地址在段内与段起点间的字节数,即地址的偏移量。,(3)距离属
25、性(也叫类型属性),58,该标号只能作段内转移,也就是说只能是与该标号所指指令同在一个逻辑段的转移指令和调用指令才能使用它。,NEAR(近):,该标号可以被非本段的转移和调用指令使用。,FAR(远):,59,当标号加在指令语句前面时,它隐含为NEAR属性。,a.隐含方式,例:SUB1:MOV AX,30H,SUB1的距离属性为NEAR也就是它只能被本段的转移指令和调用指令访问。,b.用LABEL伪指令给标号指定距离属性,类型为NEAR或FAR。该语句应与指令语句连用。,标号的距离属性可以有两种方法来指定:,60,例如:SUB1_FAR LABEL FAR SUB1:MOV AX,30H.,SU
26、B1_FAR与SUB1两个标号具有相同的段属性和偏移量属性,即相同的逻辑地址。被转移指令或调用指令访问时,是指同一个入口地址,但SUB1-FAR可以被其它段的指令调用。,61,LABEL伪指令还可以用来定义变量的属性,即改变一个变量的属性,如把字变量的高低字节作为字节变量来处理。,例如:DATA_BYTE LABEL BYTE DATA_WORD DW 20H DUP(?),DATA_BYTE与DATA_WORD具有相同的段基值和偏移量。DATA_BYTE可以被用来存取一个字节数据,而DATA_WORD则不能。,62,4.3 表达式与运算符,表达式是指令或伪指令语句操作数的常见形式。它由常数、
27、变量、标号等通过操作运算符连接而成。,注意:任何表达式的值在程序汇编过程中进行计算确定,而不是到程序运行时才计算。,8086/8088宏汇编语言中的操作运算符非常丰富,可以分为以下五类。,63,一、算术运算符,+、/、MOD、SHL、SHR、,1.运算符“+”和“-”也可作单目运算符,表示数的正负。,2.使用“+”、“-”、“”、和“/”运算符时,参加运算的数和运算结果都是整数。,64,3.“/”运算为取商的整数部分,而“MOD”运算取除法运算的余数。,65,4.SHR和SHL为逻辑移位运算符,“SHR”为右移,左边移出来的空位用0补入。“SHL”为左移,右边移出来的空位用0补入。,注意:移位
28、运算符与移位指令区别。移位运算符的操作对象是某一具体的数(常数),在汇编时完成移位操作。而移位指令是对一个寄存器或存储单元内容在程序运行时执行移位操作。,例如,NUM=11011011B.MOV AX,NUM SHL 1 MOV BX,NUM SHR 2ADD DX,NUM SHR 6,不能改成:SHL NUM,CL,66,上面的指令序列等效下面三条指令。,MOV AX,110110110BMOV BX,00110110BADD DX,3,5.下标运算符“”具有相加的作用,作用:将表达式1与表达式2的值相加后形成一个存储器操作数的地址。,下面两个语句是等效的。,MOV AX,DA_WORD20
29、HMOV AX,DA_WORD+20H,67,可以用寄存器来存放下标变量,例:下面几个语句是等价的,MOV AX,ARRAYBXSI;相对基址变址寻址MOV AX,ARRAYBX+SIMOV AX,ARRAY+BXSIMOV AX,ARRAY+SIBXMOV AX,ARRAY+BX+SI,下面是几个错误的语句。,MOV AX,ARRAY+BX+SIMOV AX,ARRAY+BXSIMOV AX,ARRAY+DA_WORD,68,二、逻辑运算符,逻辑运算符有NOT、AND、OR和XOR等四个,它们执行的都是按位逻辑运算。,69,三、关系运算符,关系运算符包括:EQ(等于)、NE(不等于)、LT(
30、小于)、LE(小于等于)、GT(大于)、GE(大于等于),关系运算符用于比较两个表达式的大小。关系运算符比较的两个表达式必须同为常数或同一逻辑段中的变量。,如果是常量的比较,则按无符号数进行比较;如果是变量的比较,则比较它们的偏移量的大小。,70,关系运算的结果只能是“真”(全1)或“假”(全0),MOV AX,0FH EQ 1111B=MOV AX,0FFFFHMOV BX,0FH NE 1111B=MOV BX,0,例1:,VAR DW NUM LT 0ABH,该语句在汇编时,根据符号常量NUM的大小来决定VAR存储单元的值,当NUM0ABH时,则变量VAR的内容为0FFFFH,否则VAR
31、的内容为0。,例2,71,四、数值返回运算符,该类运算符有5个,它们将变量或标号的某些特征值或存储单元地址的一部分提取出来。,取变量或标号所在段的段基值。,1.SEG运算符,DATA SEGMENT K1 DW 1,2 K2 DW 3,4 MOV AX,SEG K1MOV BX,SEG K2,设DATA逻辑段的段基值为1FFEH,则两条传送指令将被汇编为:,MOV AX,1FFEHMOV BX,1FFEH,作用,72,该运算符的作用是取变量或标号在段内的偏移量。,2.OFFSET运算符,DATA SEGMENT VAR1 DB 20H DUP(0)VAR2 DW 5A49H ADDR DW V
32、AR2;将VAR2的偏移量20H存入ADDR中.MOV BX,VAR2;(BX)=5A49H MOV SI,0FFSET VAR2;(SI)=20H MOV DI,ADDR;DI的内容与SI相同 MOV BP,OFFSET ADDR;(BP)=22H,73,该运算符的作用为取变量或标号的类型属性,并用数字形式来表示。对变量来说就是取它的字节长度。,3.TYPE运算符,V1 DB ABCDEV2 DW 1234H,5678HV3 DD V2;存放V2的段基值和偏移量 MOV AL,TYPE V1 MOV CL,TYPE V2 MOV CH,TYPE V3,上述程序段汇编后的等效指令序列如下:,M
33、OV AL,01HMOV CL,02HMOV CH,04H,74,该运算符只能加在变量的前面。如果变量是用重复数据操作符DUP说明的,则返回外层DUP给定的值。如果没有用DUP说明,则返回值总是1。,4.LENGTH运算符,K1 DB 10H DUP(0)K2 DB 10H,20H,30H,40HK3 DW 20H DUP(0,1,2 DUP(0)K4 DB ABCDEFGH.,MOV AL,LENGTH K1;(AL)=10HMOV BL,LENGTH K2;(BL)=1MOV CX,LENGTH K3;(CX)=20HMOV DX,LENGTH K4;(DX)=1,75,该运算符只能作用于
34、变量,SIZE取值等于LENGTH和TYPE两个运算符返回值的乘积。,5.SIZE运算符,例如,对于刚才的例子,加上以下指令:,K1 DB 10H DUP(0)K2 DB 10H,20H,30H,40HK3 DW 20H DUP(0,1,2 DUP(0)K4 DB ABCDEFGH.,MOV AL,SIZE K1;(AL)=10HMOV BL,SIZE K2;(BL)=1MOV CL,SIZE K3;(CL)=20H*2=40HMOV DL,SIZE K4;(DL)=1,76,五、属性修改运算符,这一类运算符用来对变量、标号或存储器操作数的类型属性进行修改或指定。,1.PTR运算符,类型 PT
35、R 地址表达式,使用格式:,作用:将地址表达式所指定的标号、变量或用其它形式表示的存储器地址的类型属性修改为“类型”所指的值。,类型可以是BYTE、WORD、DWORD、NEAR和FAR。这种修改是临时性的,只在含有该运算符的语句内有效。,77,DA_BYTE DB 20H DUP(0)DA_WORD DW 30H DUP(0).MOV AX,WORD PTR DA_BYTE10ADD BYTE PTR DA_WORD20,BLINC BYTE PTR BX SUB WORD PTR SI,100JMP FAR PTR SUB1;指明SUB1不是本段中的地址,78,2.HIGH/LOW运算符,
36、HIGH 表达式 LOW 表达式,使用格式:,这两个运算符用来将一个数据分离出高字节和低字节。,如果表达式为一个常量,则将其分离成高8位和低8位;如果表达式是一个地址(段基值或偏移量)时,则分离出它的高字节和低字节。,79,例如:,DATA SEGMENT CONST EQU 0ABCDH DA1 DB 10H DUP(0)DA2 DW 20H DUP(0)DATA ENDS.MOV AH,HIGH CONSTMOV AL,LOW CONSTMOV BH,HIGH(OFFSET DA1)MOV BL,LOW(OFFSET DA2)MOV CH,HIGH(SEG DA1)MOV CL,LOW(S
37、EG DA2),设DATA段的段基值是0926H,则上述指令序列汇编后的等效指令为:,MOV AH,0ABHMOV AL,0CDHMOV BH,00HMOV BL,10HMOV CH,09HMOV CL,26H,80,注意:HIGH/LOW运算符不能用来分离一个变量、寄存器或存储器单元的高字节与低字节。,下面语句使用是错误的用法。,DA1 DW 1234HMOV AH,HIGH DA1MOV BH,LOW AXMOV CH,HIGH SI,81,3.THIS运算符,THIS运算符一般与等值运算符EQU连用,用来定义一个变量或标号的类型属性。所定义的变量或标号的段基值和偏移量与紧跟其后的变量或标
38、号相同。,例如:,DATA_BYTE EQU THIS BYTEDATA_WORD DW 10 DUP(0)MOV AX,DATA_WORD MOV BL,DATA_BYTE.,82,标号LFAR与LNEAR具有相同的逻辑地址值,但类型不同。LNEAR只能被本段中的指令调用,而LFAR可以被其它段的指令调用。,83,六、运算符的优先级,在一个表达式中如果存在多个运算符时,在计算时就有先后顺序问题。不同的运算符具有不同的运算优先级别。,84,汇编程序在计算表达式时,按以下规则进行运算。,先执行优先级别高的运算,再算较低级别运算;,相同优先级别的操作,按照在表达式中的顺序,从左到右进行;,可以用圆
39、括号改变运算的顺序。,例如:,K1=10 OR 5 AND 1;结果为K1=11K2=(10 OR 5)AND 1;结果为K2=1,85,在源程序设计中,使用符号定义语句将一些常数或表达式等内容用某个指定的符号来表示。在8086/8088汇编语言中有两种符号定义语句。,4.4 符号定义语句,一、等值语句,语句格式:符号名 EQU 表达式,功能:用符合名来表示EQU右边的表达式。后面的程序中一旦出现该符合名,汇编程序将把它替换成该表达式。,一般来说,表达式可以是任何形式,常有以下几种情况。,86,1.常数或数值表达式,COUNT EQU 5NUM EQU COUNT+5,ALPHA EQU 9B
40、ETA EQU ALPHA+18BB EQU BP+8,87,2.地址表达式,ADR1被定义为在DS数据段中以BP作基址寻址的一个存储单元。,ADR1 EQU DS:BP+14,88,3.变量、寄存器名或指令助记符,例如:CREG EQU CX;在后面的程序使用CREG就是使用CX CBD EQU DAA;DAA为十进制调整指令。,注意:在同一源程序中,同一符号不能用EQU定义多次。,89,格式:符号名=表达式,二、等号语句,等号语句与等值语句具有相同的作用。但等号语句可以对一个符号进行多次定义。,例如:CONT=5NUM=14HNUM=NUM+10HCBD=DAACBD=ADD,注意:等值语
41、句与等号语句都不会为符号分配存储单元。因此所定义的符号没有段、偏移量和类型等属性。,注意:EQU语句不能重复定义,而“=”伪指令可以重复定义,其作用域从定义点到重新定义之前。,90,三、地址计数器$,地址计数器$:表示当前的偏移地址值。,例:如果用在数据段的存储单元定义中,可写成:ABC DW 1,2,$+3,4 实际存储显示:,91,例:如果用在转移指令中:JMP$+5则无条件跳转到当前指令的偏移地址+5单元继续执行。,例:如果用在数据段的存储单元定义中,ARRAY DW 1,2,$+10 实际存储显示:,92,四、置汇编计数器语句ORG,格式:ORG 数值表达式功能:将数值表达式的值赋给汇
42、编地址计数器。数值表达式的值须为065545之间的非负整数。,例:给汇编地址计数器赋值。DATASEGMENT ORG10;置值为10 VAR1DW 100H,200H ORG+5;置的值为14+5,即为19 VAR2DB 1,2,+1,+2 NEQU-VAR2;()=24DATAENDS,93,例:如果用在数据段中,该指令可以确定存储单元的偏移地址,例如将X单元的偏移地址定义为0020H,该单元的内容为5,即(DS:0020H)=5。伪指令如下:DATA SEGMENT ORG 0020H X DW 5 DATA ENDS,94,4.5、程序的段结构,8086/8088在管理内存时,需要按照
43、逻辑段进行划分,不同的逻辑段可以用来存放不同目的的数据。在程序中使用四个段寄存器CS,DS,ES和SS来访问它们。,在源程序设计时,使用伪指令来定义和使用这些逻辑段。,95,段名 SEGMENT 定位类型 组合类型 类别名.本段语句序列.段名 ENDS,一般格式:,一、段定义伪指令,伪指令SEGMENT和ENDS用于定义一个逻辑段。使用时必须配对,分别表示定义的开始与结束。,96,段名是由用户自己任意选定的,符合标识符定义规则的一个名称。,段定义伪指令语句各部分的作用如下:,1、段名,最好选用与该逻辑段用途相关的名称。如第一个数据段为DATA1,第二个数据为DATA2等。,一个段的开始与结尾用
44、的段名必须一致。,97,2、定位类型,定位类型用于决定段的起始边界,即第一个可存放数据的位置(不是段基址)。它可以有4种取值。,由于一个页面为256个字节,并且页面编号从0开始,因此,PAGE定位类型的段起始地址的最后8位二进制数一定为0,即以00H结尾的地址。,(1)PAGE:表示该段从一个页面的边界开始,(2)PARA:表示该段从一个小节的边界开始,如果用户未选定位类型,则缺省为PARA。,98,(3)WORD:表示该段从一个偶数字节地址开始,即段起始单元地址的最后一位二进制数一定是0。,(4)BYTE:表示该段起始单元地址可以是任一地址值。,注意:定位类型为PAGE和PARA时,段起始地
45、址与段基址相同。定位类型为WORD和BYTE时,段起始地址与段基址可能不同。,99,(1)NONE:若未指定组合类型,表示本段与其它段无连接关系。在装入内存时,本段有自己的物理段,因此有自己的段基址。,3、组合类型,组合类型说明符用来指定段与段之间的连接关系和定位。它有六种取值选择。,(2)PUBLIC:在满足定位类型的前提下,将与该段同名的段邻接在一起,形成一个新的逻辑段,共用一个段基址。段内的所有偏移量调整为相对于新逻辑段的段基址。,100,(3)COMMON:产生一个覆盖段。在多个模块连接时,把该段与其它也用COMMON说明的同名段置成相同的段基址,这样就达到了共享同一存储区。共享存储区
46、的长度由同名段中最大的段确定。,(4)STACK:把所有同名段连接成一个连续段,且系统自动对SS段寄存器初始化为该连续段的段基址。并初始化堆栈指针SP。,用户程序中应至少有一个段用STACK说明,否则需要用户程序自己初始化SS和SP。,101,(5)AT表达式:表示本段可定位在表达式所指示的小节边界上。表达式的值也就是段基值。,(6)MEMORY:表示本段在存储器中应定位在所有其它段这之后的最高地址上。如果有多个用MEMORY说明的段,则只处理第一个用MEMORY说明的段。其余的被视为COMMON.,102,类别名为某一个段或几个相同类型段设定的类型名称。系统在进行连接处理时,把类别名相同的段
47、存放在相邻的存储区,但段的划分与使用仍按原来的设定。,4.类别名,类别名必须用单引号引起来。所用字符串可任意选定,但它不能使用程序中的标号、变量名或其它定义的符号。,在定义一个段时,段名是必须有的项,而定位类型、组合类型和类别名三个参数是可选项。各个参数之间用空格分隔。各参数之间的顺序不能改变。,下面是一个分段结构的源程序框架。,103,STACK1 SEGMENT PARA STACK STACK0.STACK1 ENDSDATA1 SEGMENT PARA DATA.DATA1 ENDSSTACK2 SEGMENT PARA STACK0.STACK2 ENDSCODE SEGMENT P
48、ARA MEMORY ASSUME CS:CODE,DS:DATA1,SS:STACK1MAIN:.CODE ENDS END MAINDATA2 SEGMENT BYTE DATA.DATA2 ENDS,104,二、段寻址伪指令,段寻址伪指令ASSUME的作用是告诉汇编程序,在处理源程序时,定义的段与哪个寄存器关联。,ASSUME 段寄存器名:段名,段寄存器名:段名,.,一般格式:,其中段寄存器名为CS,DS,ES和SS四个之一,段名是用SEGMENT/ENDS伪指令定义的段名。,ASSUME并不设置各个段寄存器的具体内容,段寄存器的值是在程序运行时设定的。,105,DATA1 SEGMEN
49、TVAR1 DB 12HDATA1 ENDSDATA2 SEGMENT VAR2 DB 34H DATA2 ENDS CODE SEGMENT VAR3 DB 56H ASSUME CS:CODE,DS:DATA1,ES:DATA2START:.INC VAR1 INC VAR2 INC VAR3.CODE ENDS END START,该指令汇编时,VAR1使用的是DS,该指令被汇编时,VAR2使用的是ES,即指令编码中有段前缀,106,在一个代码段中可以有几条ASSUME伪指令,对于前面的设置,可以用ASSUME改变原来的设置。,一条ASSUME语句不一定设置全部段寄存器,可以选择其中一个
50、或几个段寄存器。,可以使用关键字NOTHING将前面的设置删除。,ASSUME ES:NOTHING;删除前面对ES与某个定义段的关联ASSUME NOTHING;删除全部4个段寄存器的设置,107,三、段寄存器的装入,段寄存器的初值(段基值)装入需要用程序的方法来实现。四个段寄存器的装入方法略有不同。,在程序中,使用数据传送语句来实现对DS和ES的装入。,1、DS和ES的装入,108,DATA1 SEGMENTDBYTE1 DB 12HDATA1 ENDSDATA2 SEGMENTDBYTE2 DB 14H DUP(?)DATA2 ENDSCODE SEGMENT ASSUME CS:COD