1. 实验任务1
教材「实验9 根据材料编程」(P187-189) 编程:在屏幕中间分别显示绿色、绿底红色、白底蓝色的字符串‘welcome to masm!‘。
assume cs:code, ds:data data segment db ‘welcome to masm!‘ db 2h,24h,71h data ends code segment start: mov ax,data mov ds,ax mov ax,0b800h mov es,ax mov si,1840-16 mov cx,3 mov bx,0 s: push cx mov cx,16 mov di,0 s0: mov al,ds:[di] mov ah,ds:[bx+16] mov es:[si],ax inc di inc si inc si loop s0 inc bx add si,160-32 pop cx loop s mov ah,4ch int 21h code ends end start
思路:首先在data段中以字节的形式存入字符串`welcome to masm!`,和数字 2h,24h,71h(数字中存的是要设置显示颜色的三个对应颜色含背景,详细见书189页)。代码要将`welcome to masm!`这十六个字节的数据复制到显示缓冲区中,并且显示三次,显然涉及到嵌套循环,在处理嵌套循环的时候需要注意将外循环的cx存入栈中,然后重新设置内层循环的cx,内层循环结束以后,将之前存的值出栈到cx中,实现两层循环。接下来就是把数据放入显示缓存区的操作:显示缓冲区中间的三行对应偏移地址为1760-1919,1920-2079,2080-2239,第一行的中间位置为1840,要找到放入字符的第一个位置,一共16个字符,1840前面应该放8个字符就正好能实现左右对称的效果,显示1字符需要两个字节的单元(高位颜色,低位数据),所以首位应该是1840-16。接下来就是将ds:[0~16]中的数据,和ds:[17-19]中的颜色值复制到显示缓冲区中,分别将他们存入al和ah,然后放入es:[si](显示缓冲区的位置)中即可。每次存入后si+2指向下一个显示数据的地方,bx+1寻找下一个颜色,si由1840-16,经过了16次加2,移动到了字符串的末尾,为了让位置移到下一行字符开始的位置,加上160-32。
2. 实验任务2
编写子程序printStr,实现以指定颜色在屏幕上输出字符串。调用它,完成字符串输出。
assume cs:code, ds:data data segment str db ‘try‘, 0 data ends code segment start: mov ax, data mov ds, ax mov si, offset str mov al, 2 call printStr mov ah, 4ch int 21h printStr: push bx push cx push si push di mov bx, 0b800H mov es, bx mov di, 0 s: mov cl, [si] mov ch, 0 jcxz over mov ch, al mov es:[di], cx inc si add di, 2 jmp s over: pop di pop si pop cx pop bx ret code ends end start
汇编、运行程序,观察运行结果
对源程序做如下修改: 把line3改为:str db ‘another try‘, 0,把line12改为:mov al, 4
基于运行结果,理解源代码,以及,组合使用转移指令call和ret实现子程序的原理与方法。
具体地,在 line18-40中: line19-22, line36-39,这组对称使用的push、pop,这样用的目的是什么?
push命令是为了保存之前操作过的的相关寄存器的值到栈中,pop命令是在程序相关操作结束后,将这些值出栈到对应的寄存器,和实验1中将cx入栈再出栈的原理一样。
line30的功能是什么?
将str段的数据存入低位,颜色存入高位,然后放入显示缓冲区中。
原文:https://www.cnblogs.com/lsl722/p/14152772.html