汇编语言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:code
 
code segment
    mov ax,2
    mov cx,11
s:  add ax,ax
    loop s
 
    mov ax,4c00h
    int 21h
code ends
 
end

(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命令:

用来跳过循环





  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值