实验2 多个逻辑段的汇编源程序编写与调试
实验任务1
任务1-1
对程序task1_1.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
task1_1.asm
1 assume ds:data, cs:code, ss:stack 2 data segment 3 db 16 dup(0) ; 预留16个字节单元,初始值均为0 4 data ends 5 stack segment 6 db 16 dup(0) ;预留16个字节单元,初始值均为0 7 stack ends 8 code segment 9 start: 10 mov ax, data 11 mov ds, ax 12 mov ax, stack 13 mov ss, ax 14 mov sp, 16 ; 设置栈顶 15 mov ah, 4ch 16 int 21h 17 code ends 18 end start
问题回答
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A_, 寄存器(SS) = _076B_, 寄存器(CS) = _076C_。
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__, stack的段地址是 _X-1__。
任务1-2
对程序task1_2.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
task1_2.asm
1 assume ds:data, cs:code, ss:stack 2 data segment 3 db 4 dup(0) ; 预留4个字节单元,初始值均为0 4 data ends 5 stack segment 6 db 8 dup(0) ; 预留8个字节单元,初始值均为0 7 stack ends 8 code segment 9 start: 10 mov ax, data 11 mov ds, ax 12 mov ax, stack 13 mov ss, ax 14 mov sp, 8 ; 设置栈顶 15 mov ah, 4ch 16 int 21h 17 code ends 18 end start
问题回答
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = _076A_, 寄存器(SS) = _076B_, 寄存器(CS) = _076C_。
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-2__, stack的段地址是 _X-1___。
任务1-3
对程序task1_3.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
task1_3.asm
1 assume ds:data, cs:code, ss:stack 2 data segment 3 db 20 dup(0) ; 预留20个字节单元,初始值均为0 4 data ends 5 stack segment 6 db 20 dup(0) ; 预留20个字节单元,初始值均为0 7 stack ends 8 code segment 9 start: 10 mov ax, data 11 mov ds, ax 12 mov ax, stack 13 mov ss, ax 14 mov sp, 20 ; 设置初始栈顶 15 mov ah, 4ch 16 int 21h 17 code ends 18 end start
问题回答
① 在debug中将执行到line17结束、line19之前,记录此时:寄存器(DS) = __076A__, 寄存器(SS) = __076C__, 寄存器(CS) = __076E__
② 假设程序加载后,code段的段地址是X,则,data段的段地址是__X-4__, stack的段地址是__X-2__。
任务1-4
对程序task1_4.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
task1_4.asm
1 assume ds:data, cs:code, ss:stack 2 code segment 3 start: 4 mov ax, data 5 mov ds, ax 6 mov ax, stack 7 mov ss, ax 8 mov sp, 20 9 mov ah, 4ch 10 int 21h 11 code ends 12 data segment 13 db 20 dup(0) 14 data ends 15 stack segment 16 db 20 dup(0) 17 stack ends 18 end start
问题回答
① 在debug中将执行到line9结束、line11之前,记录此时:寄存器(DS) = _076C_, 寄存器(SS) = __076E_, 寄存器(CS) = __076A__。
② 假设程序加载后,code段的段地址是X__076A__,则,data段的段地址是__X+2__, stack的段地址是 __X+4__。
任务1-5
基于上述四个实验任务的实践、观察,总结并回答:
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 __((N+15)/16)*16__。
1 xxx segment 2 db N dup(0) 3 xxx ends
② 如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令 end start 改成 end , 哪一个程序仍然可以正确执行?结合实践观察得到的结论,分析、说明原因。
task1_4.asm可以正确执行,其余的三个开始是数据块,只有4开始是程序代码。
实验任务2
编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数 据03 04。
汇编源代码
1 assume ds:data, cs:code 2 3 data segment 4 db 03,04 5 data ends 6 7 code segment 8 start: 9 mov ax, data 10 mov ds, ax 11 mov ax,0b800h 12 mov es,ax 13 mov cx,80 14 mov bx,0 15 mov si,0f00h 16 s: mov ax,[bx] 17 mov es:[si],ax 18 mov ax,[bx+1] 19 mov es:[si+1],ax 20 add si,2 21 loop s 22 23 mov ah, 4ch 24 int 21h 25 code ends 26 end start
运行结果截图
实验任务3
已知8086汇编源程序task3.asm代码片段如下。
task3.asm
1 assume cs:code 2 data1 segment 3 db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers 4 data1 ends 5 data2 segment 6 db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers 7 data2 ends 8 data3 segment 9 db 16 dup(0) 10 data3 ends 11 code segment 12 start: 13 ; ××× 14 mov ah, 4ch 15 int 21h 16 code ends 17 end start
要求:
① 编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段data3中。
补充完整的源代码
1 assume cs:code 2 data1 segment 3 db 50, 48, 50, 50, 0, 48, 49, 0, 48, 49 ; ten numbers 4 data1 ends 5 6 data2 segment 7 db 0, 0, 0, 0, 47, 0, 0, 47, 0, 0 ; ten numbers 8 data2 ends 9 10 data3 segment 11 db 16 dup(0) 12 data3 ends 13 14 code segment 15 start: 16 mov ax,data1 17 mov ds,ax 18 mov ax,data2 19 mov es,ax 20 mov ax,data3 21 mov ss,ax 22 mov bx,0 23 mov cx,10 24 s: mov ax,[bx] 25 mov ss:[bx],ax 26 mov ax,es:[bx] 27 add ss:[bx],ax 28 inc bx 29 loop s 30 mov ax,4c00h 31 int 21h 32 code ends 33 end start
② 在debug中加载、反汇编、调试。在数据项依次相加前,和相加后,分别查看三个逻辑段data1, data2, data3对应的内存空间,确认逐一相加后,结果的确保存在了逻辑段data3中。
相加之前的数据,从上往下依次为data1,data2,data3:
相加之后的数据,从上往下依次为data1,data2,data3:
显然 data3中的数据为data1与data2数据的和。
实验任务4
已知8086汇编源程序task4.asm代码片段如下。
task4.asm
1 assume cs:code 2 data1 segment 3 dw 2, 0, 4, 9, 2, 0, 1, 9 4 data1 ends 5 data2 segment 6 dw 8 dup(?) 7 data2 ends 8 code segment 9 start: 10 ; ××× 11 mov ah, 4ch 12 int 21h 13 code ends 14 end start
要求:
① 补全程序,实现把逻辑段data1中的8个字数据逆序存储到逻辑段b中。
补充完整的源代码:
1 assume cs:code 2 3 data1 segment 4 dw 2, 0, 4, 9, 2, 0, 1, 9 5 data1 ends 6 7 data2 segment 8 dw 8 dup(0) 9 data2 ends 10 11 code segment 12 start: 13 mov ax,data1 14 mov ds,ax 15 mov ax,data2 16 mov ss,ax 17 mov sp,16 18 mov cx,8 19 s: push [bx] 20 add bx,2 21 loop s 22 23 mov ah, 4ch 24 int 21h 25 code ends 26 end start
② 汇编、连接后,在debug中加载程序,运行到line15程序退出前,使用d命令查看数据段data2对应的 内存空间,确认是否实现题目要求。
加载、反汇编截图:
调试截图:
程序退出前,data2中的数据截图:
显然,数据实现了逆序存入。
实验任务5
使用任意文本编辑器,录入汇编源程序task5.asm。
task5.asm
1 assume cs:code, ds:data 2 data segment 3 db 'Nuist' 4 db 2, 3, 4, 5, 6 5 data ends 6 code segment 7 start: 8 mov ax, data 9 mov ds, ax 10 mov ax, 0b800H 11 mov es, ax 12 mov cx, 5 13 mov si, 0 14 mov di, 0f00h 15 s: mov al, [si] 16 and al, 0dfh 17 mov es:[di], al 18 mov al, [5+si] 19 mov es:[di+1], al 20 inc si 21 add di, 2 22 loop s 23 mov ah, 4ch 24 int 21h 25 code ends 26 end start
阅读源程序,从理论上分析源代码的功能,尤其是line15-25,循环实现的功能是什么,逐行理解每条指 令的功能。
对程序进行汇编、链接,得到可执行文件,运行并观察结果。
运行结果截图:
使用debug工具对程序进行调试,执行到程序返回前,即line25执行之后、line27执行之前,观察 结果。
源代码中line19的作用是?
将ds:[5+si]的数据写入es:[di+1]的位置,即字母NUIST的高位,用于染色。
修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果。
1 db 2,3,4,5,6 2 --> 改成: 3 db 5 dup(2) 或 db 5 dup(5)
结果是改变了颜色,line4的作用是控制前字符Nuist的颜色,字符Nuist位于低位,后面的数字位于高位,控制颜色输出。
实验任务6
已知8086汇编源程序task6.asm代码片段如下。
task6.asm
1 assume cs:code, ds:data 2 data segment 3 db 'Pink Floyd ' ; 16字节 4 db 'JOAN Baez ' ; 16字节 5 db 'NEIL Young ' ; 16字节 6 db 'Joan Lennon ' ; 16字节 7 data ends 8 code segment 9 start: 10 ; ××× 11 mov ah, 4ch 12 int 21h 13 code ends 14 end start
要求:
① 补全程序,将data段中的每行第一个单词从大写->小写。
补充完整的源代码:
1 assume cs:code, ds:data 2 3 data segment 4 db 'Pink Floyd ' 5 db 'JOAN Baez ' 6 db 'NEIL Young ' 7 db 'Joan Lennon ' 8 data ends 9 10 code segment 11 start: 12 mov ax,data 13 mov ds,ax 14 mov cx,2 15 mov bx,0 16 s0: mov al,[bx] 17 add al,32 18 mov [bx],al 19 add bx,48 20 loop s0 21 mov ax,data 22 add ax,1 23 mov ds,ax 24 mov cx,2 25 mov bx,0 26 s: mov al,[bx] 27 add al,32 28 mov [bx],al 29 mov al,[bx+1] 30 add al,32 31 mov [bx+1],al 32 mov al,[bx+2] 33 add al,32 34 mov [bx+2],al 35 mov al,[bx+3] 36 add al,32 37 mov [bx+3],al 38 add bx,16 39 loop s 40 41 mov ah, 4ch 42 int 21h 43 code ends 44 end start
② 在debug中加载程序,反汇编,执行到line13退出前,用d命令查看data段对应的内存空间,确认每 行第一个单词已经由大写->小写。
如图,每一行第一个单词大写均变为小写
实验任务7
问题场景描述: Power idea公司1975年-1979年的基本情况如下:
程序task7.asm的逻辑段data中(line4-6),已经定义好了这些数据。
task7.asm
1 assume cs:code, ds:data, es:table 2 data segment 3 db '1975', '1976', '1977', '1978', '1979' 4 dw 16, 22, 382, 1356, 2390 5 dw 3, 7, 9, 13, 28 6 data ends 7 table segment 8 db 5 dup( 16 dup(' ') ) ; 9 table ends 10 code segment 11 start: 12 mov ah, 4ch 13 int 21h 14 code ends 15 end start
要求:
① 补全程序,实现题目要求,把年份、收入、雇员人数、人均收入,以结构化方式写入table段中。 表格中,每行数据,在逻辑段table中占16个字节,各项数据占据字节大小分配如下。期中,数据之间用 空格间隔。
补充完整的源代码:
1 assume cs:code, ds:data, es:table 2 3 data segment 4 db '1975', '1976', '1977', '1978', '1979' 5 dw 16, 22, 382, 1356, 2390 6 dw 3, 7, 9, 13, 28 7 data ends 8 9 table segment 10 db 5 dup(16 dup(' ')) ; 11 table ends 12 13 code segment 14 start: 15 mov ax,data 16 mov ds,ax 17 mov ax,table 18 mov es,ax 19 mov bx,0 20 mov si,0 21 mov di,0 22 mov cx,5 23 s: mov ax,[bx] 24 mov es:[si],ax 25 mov ax,[bx+2] 26 mov es:[si+2],ax 27 28 mov ax,[di+20] 29 mov es:[si+5],ax 30 mov dx,0 31 mov es:[si+7],dx 32 33 div word ptr ds:[di+30] 34 mov es:[si+13],ax 35 36 mov ax,[di+30] 37 mov es:[si+10],ax 38 39 add di,2 40 add bx,4 41 add si,16 42 loop s 43 44 mov ah, 4ch 45 int 21h 46 code ends 47 end start
② 汇编、连接后,在debug中加载、调试程序。灵活、合理使用u命令、g命令、d命令,显示刚开始逻辑段table的数据信息,以及,结构化存入数据后,数据段table的数据信息,确认实现题目要求。
程序加载后的初始数据:
程序退出前,table中的数据:
如图,结构化存入数据,之间用空格隔开
实验总结
1、ax赋值显存段地址时,需要输入0b800h,否则会编译出错。
2、程序加载后分配空间是以16个字节为单位的,也就是说如果不足16个字节的也分配16个字节。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具