1. 实验任务1
任务1-1
task1_1.asm源码
assume ds:data, cs:code, ss:stack data segment db 16 dup(0) data ends stack segment db 16 dup(0) stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 16 mov ah, 4ch int 21h code ends end start
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A___, 寄存器(SS) = _076B___, 寄存器(CS) = _076C___
② 假设程序加载后,code段的段地址是X,则,data段的段地址是_X-2___, stack的段地址是_X-1___。
任务1-2
task1_2.asm源码
assume ds:data, cs:code, ss:stack data segment db 4 dup(0) data ends stack segment db 8 dup(0) stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 8 mov ah, 4ch int 21h code ends end start
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A___, 寄存器(SS) = _076B___, 寄存器(CS) = _076C___
② 假设程序加载后,code段的段地址是X,则,data段的段地址是_X-2___, stack的段地址是_X-1___。
因为程序占用的内存空间是连续的,段大小占16字节的整数倍,故4和8字节的段同样会占用16字节,所以结果和task1_1一致。
任务1-3
assume ds:data, cs:code, ss:stack data segment db 20 dup(0) data ends stack segment db 20 dup(0) stack ends code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 20 mov ah, 4ch int 21h code ends end start
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A___, 寄存器(SS) = _076C___, 寄存器(CS) = _076E___
② 假设程序加载后,code段的段地址是X,则,data段的段地址是_X-4___, stack的段地址是_X-2___。
因为程序占用的内存空间是连续的,段大小占16字节的整数倍,故20字节的段同样会占用32字节,所以结果是这样。
任务1-4任务
task1_4.asm源码
assume ds:data, cs:code, ss:stack code segment start: mov ax, data mov ds, ax mov ax, stack mov ss, ax mov sp, 20 mov ah, 4ch int 21h code ends data segment db 20 dup(0) data ends stack segment db 20 dup(0) stack ends end start
① 在debug中将执行到line9结束、line11之前,记录此时:寄存器(DS) = _076C___, 寄存器(SS) = _076E___, 寄存器(CS) = _076A___
② 假设程序加载后,code段的段地址是X,则,data段的段地址是_X+2___, stack的段地址是_X+4___。
因为程序占用的内存空间是连续的,段大小占16字节的整数倍,故20字节的段同样会占用32字节,又因为代码段数据段栈段顺序有变化,所以结果是这样。
任务1-5
xxx segment db N dup(0) xxx ends
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 _ceil(N/16)*16___。
② 如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令end start改成end, 哪一个程序仍然可以正确执行。结合实践观察得到的结论,分析、说明原因。
task1_1.asm, task1_2.asm, task1_3.asm三者不能执行,因为它们都是数据段开头,将end start替换成end意味着程序未指定开始位置,会从开头开始执行,而执行数据段中的数据自然会出错。只有task1_4.asm可以执行,因为它开头即是代码段。
2. 实验任务2
汇编源代码
assume cs:code code segment start: mov ax,0b800h mov ds,ax mov bx,0f00h mov cx,80 s: mov [bx],0403h inc bx inc bx loop s mov ah,4ch int 21h code ends end start
运行结果截图
3. 实验任务3
补充完整后的汇编源代码
assume cs:code data1 segment db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers data1 ends data2 segment db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers data2 ends data3 segment db 16 dup(0) data3 ends code segment start: mov cx,10 mov bx,0 mov dx,0 s: mov ax,data1 mov ds,ax mov dl,[bx] mov ax,data2 mov ds,ax add dl,[bx] mov ax,data3 mov ds,ax mov [bx],dl inc bx loop s mov ah,4ch int 21h code ends end start
在debug中加载、反汇编、调试截图
在数据项依次相加之前,查看逻辑段data1, data2, data3对应的内存空间数据原始值的debug命令和截图
依次相加之后,查看逻辑段data1, data2, data3对应的内存空间数据原始值的debug命令和截图
4. 实验任务4
补充完整后的汇编源代码
assume cs:code data1 segment dw 2, 0, 4, 9, 2, 0, 1, 9 data1 ends data2 segment dw 8 dup(?) data2 ends code segment start: mov ax,data1 mov ds,ax mov ax,data2 mov ss,ax mov sp,16 mov cx,8 mov bx,0 s: push [bx] add bx,2 loop s mov ah, 4ch int 21h code ends end start
在debug中加载、反汇编、调试截图
在程序退出前,使用d命令查看数据段data2对应的内存空间的截图。
5. 实验任务5
task5.asm源码
assume cs:code, ds:data data segment db 'Nuist' db 2, 3, 4, 5, 6 data ends code segment start: mov ax, data mov ds, ax mov ax, 0b800H mov es, ax mov cx, 5 mov si, 0 mov di, 0f00h s: mov al, [si] and al, 0dfh mov es:[di], al mov al, [5+si] mov es:[di+1], al inc si add di, 2 loop s mov ah, 4ch int 21h code ends end start
运行结果截图
使用debug工具对程序进行调试,使用g命令一次性执行到程序返回前(即ine25执行之后、line27执行之前)的截图
源代码中line19的作用是?
答:将小写字母转化成大写字母。
and al, 0dfh其中0dfh转化成二进制是1101 1111(B),
大写字母ASCII码为0100 0001~0101 1010,
小写字母ASCII码为0110 0001~0111 1010,
小写字母的ASCII码与odfh与之后,第三位的1会变成0,也就从小写字母转化成了大写字母。
源代码中data段line4的字节数据的用途是?
答:设置显示内容的颜色。
改变第四行:
6. 实验任务6
task6.asm源代码
assume cs:code, ds:data data segment db 'Pink Floyd ' db 'JOAN Baez ' db 'NEIL Young ' db 'Joan Lennon ' data ends code segment start: mov ax,data mov ds,ax mov cx,4 mov bx,0 s: mov al,[bx] or al,20h mov [bx],al add bx,16 loop s mov ah, 4ch int 21h code ends end start
在debug中加载、反汇编、调试截图
在程序退出前,使用d命令查看数据段data对应的内存空间的截图。
7. 实验任务7
task7.asm源码
assume cs:code, ds:data, es:table data segment db '1975', '1976', '1977', '1978', '1979' dw 16, 22, 382, 1356, 2390 dw 3, 7, 9, 13, 28 data ends table segment db 5 dup( 16 dup(' ') ) ; table ends code segment start: mov ax,data mov ds,ax mov ax,table mov es,ax mov cx,5 mov bx,0 mov si,0 mov di,0 mov dx,0 s1: mov ax,[bx] mov es:[si],ax mov ax,[bx+2] mov es:[si+2],ax add bx,4 add si,16 loop s1 ; mov bx,20 mov si,5 mov cx,5 s2: mov ax,[bx] mov es:[si],ax mov ax,0 mov es:[si+2],ax add bx,2 add si,16 loop s2 ;mov bx,30 mov si,10 mov cx,5 s3: mov ax,[bx] mov es:[si],ax add bx,2 add si,16 loop s3 mov si,0 mov cx,5 s4: mov ax, es:[si+5] mov dx, es:[si+7] div word ptr es:[si+10] mov es:[si+13],ax add si,16 loop s4 mov ah, 4ch int 21h code ends end start
查看table段原始数据信息截图
在debug中运行到程序退出之前,使用d命令查看table段对应的内存空间的截图,确认信息是否按要求结构化地写入到指定内存
五、实验总结
1.or 20h和and 0dfh两条命令非常实用,前者可以实现大写字母转小写,后者可以实现小写字母转大写,二者结合灵活运用,可以轻松实现大小写字母的混合格式输出。
2.为了防止调用某个寄存器值时被未知的数据错误”污染“寄存器,在赋值阶段,建议将需要使用的却又不赋具体值的寄存器先置0。