实验2 多个逻辑段的汇编源程序编写与调试
1. 实验任务1
此实验任务中,包含4个子任务。逐一实践,结合实践观察、验证,回答问题。
任务1-1
对程序task1_1.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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) = ____, 寄存器(SS) =
____, 寄存器(CS) = ____
使用U命令查看反汇编代码,找到mov ah,4ch这一代码的内存地址,并使用G命令快速执行到这条命令之前。

(DS)=076A
(SS)=076B
(CS)=076C
② 假设程序加载后,code段的段地址是X,则,data段的段地址是____, stack的段地址是____。
假设程序加载后,CODE段地址为X,则data段地址为X-2,stack段地址为X-1
任务1-2
对程序task1_2.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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) = ____, 寄存器(SS) =
____, 寄存器(CS) = ____
使用u命令查看反汇编代码,找到mov ah,4ch这一代码的内存地址,并使用g命令快速执行到这条命令之前。

(DS)=076A
(SS)=076B
(CS)=076C
② 假设程序加载后,code段的段地址是X,则,data段的段地址是____, stack的段地址是____。
假设程序加载后,CODE段地址为X,则data段地址为X-2,stack段地址为X-1。
任务1-3
对程序task1_3.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
task1_3.asm
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) = ____, 寄存器(SS) =
____, 寄存器(CS) = ____
使用u命令查看反汇编代码,找到mov ah,4ch这一代码的内存地址,并使用g命令快速执行到这条命令之前。

(DS)=076A
(SS)=076C
(CS)=076E
② 假设程序加载后,code段的段地址是X,则,data段的段地址是____, stack的段地址是____。
假设程序加载后,CODE段地址为X,则data段地址为X-4,stack段地址为X-2。
任务1-4
对程序task1_4.asm进行汇编、连接,用debug加载、跟踪调试,基于结果,回答问题。
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中将执行到line17结束、line19之前,记录此时:寄存器(DS) = ____, 寄存器(SS) =
____, 寄存器(CS) = ____
使用u命令查看反汇编代码,找到mov ah,4ch这一代码的内存地址,并使用g命令快速执行到这条命令之前。

(DS)=076C
(SS)=076E
(CS)=076A
② 假设程序加载后,code段的段地址是X,则,data段的段地址是____, stack的段地址是____。
假设程序加载后,CODE段地址为X,则data段地址为X+2,stack段地址为X+4。
任务1-5
基于上述四个实验任务的实践、观察,总结并回答:
① 对于如下定义的段,程序加载后,实际分配给该段的内存空间大小是 ____。
xxx segment
db N dup(0)
xxx ends
当本段不是程序的最后一段时: ⌈N/16⌉×16。
当本段是程序的最后一段时: N
② 如果将程序task1_1.asm, task1_2.asm, task1_3.asm, task1_4.asm中,伪指令 end start 改成
end , 哪一个程序仍然可以正确执行?结合实践观察得到的结论,分析、说明原因。
程序task1_1.asm

不能正常运行
程序task1_2.asm

不能正常运行
程序task1_3.asm

不能正常运行
程序task1_4.asm

可以正确执行
task1_1、task1_2、task1_3不能正确执行,task1_4可以正确执行。
汇编器在处理汇编源程序的时候会先处理伪指令。start伪指令表示程序段从此处开始,会将code segment给CS段寄存器。由于start和end start需要成对出现,汇编器找到start后,却没找到end start,所以start无效,因此从程序段前缀之后开始执行程序,由于1,2,3个子任务的程序段前缀之后都是数据段,因为在8086汇编环境下,数据和指令处于同等地位,所以CPU无法辨别内存中的是指令还是数据,所以1,2,3个子任务不能正确完成期望的代码的执行。
2. 实验任务2
编写一个汇编源程序,实现向内存单元b800:0f00 ~ b800:0f9f连续160字节,依次重复填充十六进制数
据03 04
assume cs:code
code segment
start:
mov ax,0b800h
mov ds,ax
mov bx,0f00h
mov cx,80
p:
mov ds:[bx],0403h
inc bx
inc bx
loop p
mov ah,4ch
int 21h
code ends
end start
运行结果如下

3. 实验任务3
已知8086汇编源程序task3.asm代码片段如下。
task3.asm
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:
; ×××
code ends
end start
要求:
① 编程实现把逻辑段data1和逻辑段data2的数据依次相加,结果保存到逻辑段data3中。
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 ax,data1
mov ds,ax
mov cx,16
mov bx,0
s:
mov al,ds:[bx]
add al,ds:[bx+16]
mov ds:[bx+32],al
inc bx
loop s
mov ah,4ch
int 21h
code ends
end start
② 在debug中加载、反汇编、调试。在数据项依次相加前,和相加后,分别查看三个逻辑段data1,
data2, data3对应的内存空间,确认逐一相加后,结果的确保存在了逻辑段data3中。
程序运行结果如下

4. 实验任务4
已知8086汇编源程序task4.asm代码片段如下。
task4.asm
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 ah, 4ch
int 21h
code ends
end start
要求:
① 补全程序,实现把逻辑段data1中的8个字数据逆序存储到逻辑段b中。
assume cs:code
data1 segment
dw 2, 0, 4, 9, 2, 0, 1, 9
data1 ends
data2 segment
dw 8 dup(?)
data2 ends
stack segment
db 16 dup(0)
stack ends
code segment
start:
mov ax,data1
mov ds,ax
mov ax,stack
mov ss,ax
mov sp,16
mov bx,0
mov cx,8
s1:
push ds:[bx]
inc bx
inc bx
loop s1
mov bx,16
mov cx,8
s2:
pop ds:[bx]
inc bx
inc bx
loop s2
mov ah, 4ch
int 21h
code ends
end start
② 汇编、连接后,在debug中加载程序,运行到line15程序退出前,使用d命令查看数据段data2对应的
内存空间,确认是否实现题目要求。

5. 实验任务5
使用任意文本编辑器,录入汇编源程序task5.asm。
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
阅读源程序,从理论上分析源代码的功能,尤其是line15-25,循环实现的功能是什么,逐行理解每条指
令的功能。
对程序进行汇编、链接,得到可执行文件,运行并观察结果。

使用debug工具对程序进行调试,执行到程序返回前,即line25执行之后、line27执行之前,观察
结果。

源代码中line19的作用是?
and 0dfh是将寄存器中数据和0dfh进行与运算,将0dfh转化为二进制数为11011111,进行与运算后ascall差值为32,实现将小写字母转化为大写字母。
修改line4里5个字节单元的值,重新汇编、链接、运行,观察结果。
db 2,3,4,5,6 --> 改成: db 5 dup(2) 或 db 5 dup(5)
基于观察,分析、猜测这里的数值作用是什么。

数值的作用是对字母颜色进行修改
6. 实验任务6
已知8086汇编源程序task6.asm代码片段如下。
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 ah, 4ch
int 21h
code ends
end start
要求:
① 补全程序,将data段中的每行第一个单词从大写->小写。
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 dx,cx
mov cx,4
mov si,0
p:
mov al,ds:[bx+si]
or al,020h
mov ds:[bx+si],al
inc si
loop p
add bx,16
mov cx,dx
loop s
mov ah, 4ch
int 21h
code ends
end start
② 在debug中加载程序,反汇编,执行到line13退出前,用d命令查看data段对应的内存空间,确认每
行第一个单词已经由大写->小写。

7. 实验任务7
问题场景描述:
Power idea公司1975年-1979年的基本情况如下:
程序task7.asm的逻辑段data中(line4-6),已经定义好了这些数据。
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 ah, 4ch
int 21h
code ends
end start
要求:
① 补全程序,实现题目要求,把年份、收入、雇员人数、人均收入,以结构化方式写入table段中。
表格中,每行数据,在逻辑段table中占16个字节,各项数据占据字节大小分配如下。期中,数据之间用
空格间隔。
assume cs:codesg
data segment
db '1975', '1976', '1977', '1978', '1979'
dd 16, 22, 382, 1356, 2390
dw 3, 7, 9, 13, 28
data ends
table segment
db 5 dup( 16 dup(' ') )
table ends
codesg segment
start:
mov ax, data
mov ds, ax
mov ax, table
mov es, ax
mov bx, 0
mov bp, 0
mov cx, 5
lay_years:
mov dx, cx
mov cx, 4
mov si, 0
lay_year:
mov al, ds:[bx+si]
mov es:[bp+si], al
inc si
loop lay_year
mov cx, dx
mov byte ptr es:[bp+4],' '
add bx, 4
add bp, 10h
loop lay_years
mov bp, 0
mov cx, 5
lay_incomes:
mov ax,ds:[bx]
mov es:[bp+5],ax
mov ax,ds:[bx+2]
mov es:[bp+7],ax
mov byte ptr es:[bp+9], ' '
add bx, 4
add bp, 10h
loop lay_incomes
mov bp, 0
mov cx, 5
lay_populations:
mov ax, ds:[bx]
mov es:[bp+0ah], ax
mov byte ptr es:[bp+0ch], ' '
add bx, 2
add bp, 10h
loop lay_populations
mov bp,0
mov cx,5
lay_average_incomes:
mov ax, es:[bp+5]
mov dx, es:[bp+7]
div word ptr es:[bp+0ah]
mov es:[bp+0dh], ax
mov byte ptr es:[bp+0fh], ' '
add bp, 10h
loop lay_average_incomes
mov ax, 4c00h
int 21h
codesg ends
end start
② 汇编、连接后,在debug中加载、调试程序。灵活、合理使用u命令、g命令、d命令,显示刚开始逻
辑段table的数据信息,以及,结构化存入数据后,数据段table的数据信息,确认实现题目要求。

实验总结
1、可以将整个数据段地址存入段地址寄存器中,在通过偏移地址实现对段内数据的访问。
2、end start除了通知编译器程序结束,也是指出程序的入口地址。
3、在实现大小写字母相互转化时,可以根据字符的Ascall码值,通过或运算、与运算实现转化,提高运算效率。
4、8086的显存在存放在显示内容时,以一个字为单位,低字节为打印的字符内容,高字节为8位色彩代码。

浙公网安备 33010602011771号