本文主要是介绍汇编语言04——[BX]和loop指令,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
整理自fishcc论坛课件
首先展示一个新的程序:
assume cs:codesg
codesg segment
start: mov ax,2000H
mov ds,ax
mov al,[0]
mov bl,[1]
mov cl,[2]
mov dl,[3]
mov ax,4C00H
int 21H
codesg ends
end start
使用masm进行编译连接
这里,debug中对于语句的解释和masm中不同
在masm进行编译时,mov al,[0]被解释为将0放入al,和mov al,0等同
但是,当在debug中直接写入代码时,mov al,[0]被解释为将段地址为ds中的值,偏移地址为0的地址中的数据放入al
在masm中,使用[bx]来表示一个内存单元:bx中的偏移地址和ds中储存的段地址组成的地址
loop
描述性符号“()”
为了描述上的简洁,在以后的课程中,我们将使用一个描述性的符号 “() ”来表示一个寄存器或一个内存单元中的内容。
我们看一下(X)的应用,比如:
(1)ax中的内容为0010H,我们可以这样来描述:(ax)=0010H;
(2)2000:1000 处的内容为0010H,我们可以这样来描述:(21000H)=0010H;
(3)对于mov ax,[2]的功能,我们可以这样来描述:(ax)=((ds)*16+2);
(4)对于mov [2],ax 的功能,我们可以这样来描述:((ds)*16+2)=(ax);
(5)对于 add ax,2 的功能,我们可以这样来描述:(ax)=(ax)+2;
(6)对于add ax,bx的功能,我们可以这样来描述:(ax)=(ax)+(bx);
(7)对于push ax的功能,我们可以这样来描述:
(sp) = (sp)-2
((ss)*16+(sp))=(ax)
(8)对于pop ax 的功能,我们可以这样来描述:
(ax)=((ss)*16+(sp))
(sp)=(sp)+2
()可以理解为C语言中的指针取值的*
约定符号idata表示常量
我们在Debug 中写过类似的指令:mov ax,[0],表示将 ds:0 处的数据送入ax中。指令中,在“[…]”里用一个常量0表示内存单元的偏移地址。
以后,我们用idata表示常量。
例如:
mov ax,[idata] 就代表 mov ax,[1]、mov ax,[2]、mov ax,[3] 等。
mov bx,idata 就代表 mov bx,l、mov bx,2、mov bx,3 等。
mov ds,idata 就代表 mov ds,1、mov ds,2 等,它们都是非法指令。
Loop指令
指令的格式是:loop 标号,CPU 执行loop指令的时候,要进行两步操作:
(cx)=(cx)-1;
判断cx中的值,不为零则转至标号处执行程序,如果为零则向下执行。
从上面的描述中,我们可以看到,cx中的值影响着loop指令的执行结果。
通常(注意,我们说的是通常)我们用loop指令来实现循环功能,cx 中存放循环次数。
assume cs:codecode segmentmov ax,2mov cx,11
s: add ax,axloop smov ax,4c00hint 21h
code endsend
(1)标号在汇编语言中,标号代表一个地址,此程序中有一个标号s 。它实际上标识了一个地址,这个地址处有一条指令:add ax,ax。
(2)loop s
CPU 执行loop s的时候,要进行两步操作:
(cx)=(cx)-1;
判断cx 中的值,不为0 则转至标号s 所标识的地址处执行(这里的指令是“add ax,ax),如果为零则执行下一条指令(下一条指令是mov ax,4c00h)。
(3)以下三条指令
mov cx,11
s: add ax,ax
loop s
执行loop s时,首先要将(cx)减1,然后若(cx)不为0,则向前转至s处执行add ax,ax。所以,我们可以利用cx来控制add ax,ax的执行次数。
总结:
(1)在cx中存放循环次数;
(2)loop 指令中的标号所标识地址要在前面;
(3)要循环执行的程序段,要写在标号和loop 指令的中间。
用cx和loop指令相配合实现循环功能的程序框架如下:
mov cx,循环次数
s: 循环执行的程序段
loop s
在Debug中跟踪用loop指令实现的循环程序
考虑这样一个问题,计算ffff:0006单元中的数乘以3,结果存储在dx中。我们分析一下:
(1)运算后的结果是否会超出dx所能存储的范围?
ffff:0006 单元中的数是一个字节型的数据,范围在0~255之间,则用它和3相乘结果不会大于65535,可以在dx 中存放下。
(2)我们用循环累加来实现乘法,用哪个寄存器进行累加?
我们将ffff:0006单元中的数赋值给ax,用dx进行累加。先设(dx)=0,然后做3次(dx)=(dx)+(ax)。
(3) ffff:0006单元是一个字节单元,ax是一个 16 位寄存器,数据长度不一样,如何赋值?
注意,我们说的是“赋值”,就是说,让 ax 中的数据的值(数据的大小)和ffff:0006 单元中的数据的值(数据的大小)相等。
8位数据01H和16位数据0001H的数据长度不一样,但它们的值是相等的。
那么我们如何赋值?
ffff:0006单元中的数据是XXH,若要ax中的值和ffff:0006单元中的相等,ax中的数据应为00XXH。
所以,若实现ffff:0006单元向ax 赋值,我们应该令(ah)=0,(al)=(ffff6H)。
实现计算ffff:0006单元中的数乘以3,结果存储在dx中的程序代码。
注意程序中的第一条指令mov ax,0ffffh。
我们知道大于9FFFH的十六进制数据A000H、A001H、…… 、C000H、C001H、……、FFFEH、FFFFH等,在书写的时候都是以字母开头的。而在汇编源程序中,数据不能以字母开头,所以要在前面加0。
debug中的p命令:
用来结束程序,用来结束循环
debug中的g命令:
用来跳过循环
这篇关于汇编语言04——[BX]和loop指令的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!