实验8汇编实验

  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

实验八编写子程序(4学时)

在本次实验中,我们将编写3个子程序,通过它们来认识几个常见的问题和掌握解决这些问题的方法。同前面的所有实验一样,这个实验室必须独立完成的,在后面的课程中,将要用到这个实验中编写的3个子程序。

1、显示字符串

显示字符串是下现实工作中经常要用到的功能,应该编写一个通用的字程序来实现这个功能。我们应该提供灵活的调用接口,使调用者可以决定显示的位置(行、列)、内容和颜色。

该子程序描述如下:

名称:show_str

功能:在指定位置用指定颜色显示一个用0结束的字符串。

参数:(dh)=行号(取值范围0~24),(dl)=列号(取值范围0~79),(cl)=颜色,ds:si 指向字符串的首地址。

返回:无

应用举例:在屏幕的8行3列,用绿色显示data段中的字符串。

assume cs: code

data segment

db ‘Welcome to masm!’, 0

data ends

code segment

start: mov dh, 8

mov dl, 3

mov cl, 2

mov ax, data

mov ds, ax

mov si, 0

call show_str

mov ax, 4c00h

int 21h

show_str: :

:

:

code ends

end start

提示:

(1)子程序的入口参数是屏幕上的行号和列号,注意在子程序内部要将它们转化为显存中的地址,首先要分析一下屏幕上的行列位置和显存地址的对应关系;

(2)注意保存子程序中用到的相关寄存器;

(3)这个子程序的内部处理和显存的结构密切相关,但是向外提供了与显存结构无关的接口。通过调用这个子程序,进行字符串的显示时可以不必了解显存的结果,为编程提供了方便。在实验中,注意体会这种设计思想。

data segment

db "Welcome to masm!" data ends

stack segment

db "Welcome to masm!" stack ends

code segment

start:

mov dh, 8

mov dl, 3

mov cl, 2

mov ax, data

mov ds, ax

mov si, 0

mov ax,0B800h

mov es,ax

mov ax, stack

mov ss, ax

mov sp,10h

call show_str

mov ax, 4c00h

int 21h

show_str:

push dx

push cx

mov al,160

dec dh

mul dh

mov bx,ax

add dx,si

add dl,dl

add bl,dl

mov cl,[si]

mov ch,0

jcxz ok

mov es:[bx],cl

pop cx

mov es:[bx+1],cl

inc si

pop dx

jmp short show_str

ok:

pop cx

pop dx

ret

code ends

end start

2、解决除法溢出的问题

前面讲过,div指令可以做除法。当进行8位除法的时候,用al存储商,ah存储余数;进行16位除法时,用ax存储商,dx存储余数。可是,现在有一个问题,如果结果的商大于al或ax所能存储的最大值,那么将如何?

比如,下面的程序段:

mov bh, 1

mov ax, 1000

div bh

进行的是8位除法,商为1000,而1000在al中放不下。

又比如,下面的程序段:

mov ax, 1000H

mov dx, 1

mov bx, 1

div bx

进行的是16位除法,商为11000H,而11000H在ax中放不下。

我们在用div指令做除法的时候,和可能发生上面的情况:结果的商过大,超出了寄存器所能存储的范围。当CPU执行div等除法指令时,如果发生这样的情况,将引发CPU的一个内部错误,这个错误被称为:除法溢出。我们可以通过特殊的程序来处理这个错误,但在这里我们不讨论这个错误的处理,这是后面课程中要涉及的内容。下面我们仅仅来看一下

除法发生时的一些现象,如同所示。

图除法溢出时发生的现象

图中展示了在Windows 2000中使用Debug执行相关程序段的结果,div指令引发了CPU的除法溢出,系统对其进行了相关的处理。

至此,我们已经清楚了问题所在;用div指令做除法的时候可能产生除法溢出。由于有这样的问题,在进行除法运算时要注意除数和被除数的值,比如1000000/10就不能用div指令来计算。那么怎么办呢?我们用下面的子程序divdw解决。

该子程序的描述如下:

名称:divdw

功能:进行不会产生溢出的除法运算,被除数为dword型,除数为word型,结果为dword型。

参数: (ax)=dword型数据的低16位;(dx)=dword型数据的高16位;(cx)=除数返回:(ax)=商的低16位;(dx)=商的高16位;(cx)=余数

应用举例:计算1000000/10(F424H/0AH)

mov ax, 4240H

mov dx, 000FH

mov cx, 0AH

call divdw

结果:(ax)=86A0H,(dx)=0001H, (cx)=0.

提示:

给出一个公式:

X: 被除数,范围:[0,FFFFFFFF]

N: 除数,范围: [0,FFFF]

H: X高16位,范围: [0,FFFF]

L: X低16位,范围: [0,FFFF]

int(): 描述性运算符,取商,比如int(38/10)=3

rem(): 描述性运算符,取余数,比如rem(38/10)=8

公式:X/N=int(H/N)*65536+[rem(H/N)*65536+L]/N

相关文档
最新文档