由于我自己的没法复制,并且不带的更改了,所以可能会和其他人有些答案重合,但确保正确。
(1)将下面的程序编译连接,用Debug加载、跟踪,然后回答问题。
assume cs:code,ds:data,ss:stack
data segment
dw 0123h,0456h,0789h,0abch,0defh,0fedh,0cbah,0987h
data ends
stack segment
dw 0,0,0,0,0,0,0,0
stack ends
code segment
start: mov ax,stack
mov ss,ax
mov sp,16
mov ax,data
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
end start
①CPU执行程序,程序返回前,data段中的数据 不变 。
②CPU执行程序,程序返回前,CS= 0C88H ,SS= 0C87H ,DS= 0C86H 。
③设程序加载后,CODE段的段地址为X,则DATA段的段地址为 X-2 ,STACK段的段地址为 X-1 。
C:\DOCUME~1\ADMINI~1>debug sy5-1.exe
-r
AX=0000 BX=0000 CX=0042 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000
DS=0C76 ES=0C76 SS=0C86 CS=0C88 IP=0000 NV UP EI PL NZ NA PO NC
0C88:0000 B8870C MOV AX,0C87
-d 0c86:0 f
0C86:0000 23 01 56 04 89 07 BC 0A-EF 0D ED 0F BA 0C 87 09 #.V.............
-u
0C88:0000 B8870C MOV AX,0C87
0C88:0003 8ED0 MOV SS,AX
0C88:0005 BC1000 MOV SP,0010
0C88:0008 B8860C MOV AX,0C86
0C88:000B 8ED8 MOV DS,AX
0C88:000D FF360000 PUSH [0000]
0C88:0011 FF360200 PUSH [0002]
0C88:0015 8F060200 POP [0002]
0C88:0019 8F060000 POP [0000]
0C88:001D B8004C MOV AX,4C00
-g 001d
AX=0C86 BX=0000 CX=0042 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000
DS=0C86 ES=0C76 SS=0C87 CS=0C88 IP=001D NV UP EI PL NZ NA PO NC
0C88:001D B8004C MOV AX,4C00
-d 0c86:0 f
0C86:0000 23 01 56 04 89 07 BC 0A-EF 0D ED 0F BA 0C 87 09 #.V.............
-q
(2)将下面的程序编译连接,用Debug加载、跟踪,然后回答问题。
assume cs:code,ds:data,ss:stack
data segment
dw 0123h,0456h
data ends
stack segment
dw 0,0
stack ends
code segment
start: mov ax,stack
mov ss,ax
mov sp,16
mov ax,data
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
end start
①CPU执行程序,程序返回前,data段中的数据 不变 。
②CPU执行程序,程序返回前,CS= 0C88H ,SS= 0C87H ,DS= 0C86H 。
③设程序加载后,CODE段的段地址为X,则DATA段的段地址为 X-2 ,STACK段的段地址为 X-1 。
④对于如下定义的段:
name segment
……
name ends
如果段中的数据占N个字节,则程序加载后,该段实际占有的空间为 ((N+15)/16)*16 。
④解析:
N分为被16整除和不被16整除。
当N被16整除时: 占有的空间为(N/16)*16
当N不被16整除时: 占有的空间为(N/16+1)*16,N/16得出的是可以整除的部分,还有一个余数,余数肯定小于16,加上一个16。
程序加载后分配空间是以16个字节为单位的,也就是说如果不足16个字节的也分配16个字节。
两种情况总结成一个通用的公式:((N+15)/16)*16
C:\DOCUME~1\ADMINI~1>debug sy5-2.exe
-r
AX=0000 BX=0000 CX=0042 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000
DS=0C76 ES=0C76 SS=0C86 CS=0C88 IP=0000 NV UP EI PL NZ NA PO NC
0C88:0000 B8870C MOV AX,0C87
-d 0c86:0 3
0C86:0000 23 01 56 04 #.V.
-u
0C88:0000 B8870C MOV AX,0C87
0C88:0003 8ED0 MOV SS,AX
0C88:0005 BC1000 MOV SP,0010
0C88:0008 B8860C MOV AX,0C86
0C88:000B 8ED8 MOV DS,AX
0C88:000D FF360000 PUSH [0000]
0C88:0011 FF360200 PUSH [0002]
0C88:0015 8F060200 POP [0002]
0C88:0019 8F060000 POP [0000]
0C88:001D B8004C MOV AX,4C00
-g 001d
AX=0C86 BX=0000 CX=0042 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000
DS=0C86 ES=0C76 SS=0C87 CS=0C88 IP=001D NV UP EI PL NZ NA PO NC
0C88:001D B8004C MOV AX,4C00
-d 0c86:0 3
0C86:0000 23 01 56 04 #.V.
-q
(3)将下面的程序编译连接,用Debug加载、跟踪,然后回答问题。
assume cs:code,ds:data,ss:stack
code segment
start: mov ax,stack
mov ss,ax
mov sp,16
mov ax,data
mov ds,ax
push ds:[0]
push ds:[2]
pop ds:[2]
pop ds:[0]
mov ax,4c00h
int 21h
code ends
data segment
dw 0123h,0456h
data ends
stack segment
dw 0,0
stack ends
end start
①CPU执行程序,程序返回前,data段中的数据 不变 。
②CPU执行程序,程序返回前,CS= 0C86H ,SS= 0C8AH ,DS= 0C89H 。
③设程序加载后,CODE段的段地址为X,则DATA段的段地址为 X+3 ,STACK段的段地址为 X+4 。
C:\DOCUME~1\ADMINI~1>debug sj5-3.exe
-r
AX=0000 BX=0000 CX=0044 DX=0000 SP=0000 BP=0000 SI=0000 DI=0000
DS=0C76 ES=0C76 SS=0C86 CS=0C86 IP=0000 NV UP EI PL NZ NA PO NC
0C86:0000 B88A0C MOV AX,0C8A
-u
0C86:0000 B88A0C MOV AX,0C8A
0C86:0003 8ED0 MOV SS,AX
0C86:0005 BC1000 MOV SP,0010
0C86:0008 B8890C MOV AX,0C89
0C86:000B 8ED8 MOV DS,AX
0C86:000D FF360000 PUSH [0000]
0C86:0011 FF360200 PUSH [0002]
0C86:0015 8F060200 POP [0002]
0C86:0019 8F060000 POP [0000]
0C86:001D B8004C MOV AX,4C00
-g 001d
AX=0C89 BX=0000 CX=0044 DX=0000 SP=0010 BP=0000 SI=0000 DI=0000
DS=0C89 ES=0C76 SS=0C8A CS=0C86 IP=001D NV UP EI PL NZ NA PO NC
0C86:001D B8004C MOV AX,4C00
-d 0c89:0 3
0C89:0000 23 01 56 04 #.V.
-q
(4)如果将(1)、(2)、(3)题中的最后一条伪指令“end start”改为“end”(也就是说,不指明程序的入口),则哪个程序仍然可以正确执行?请说明原因。
答:第三条程序仍然可以正确执行,如果不指明入口位置,则程序从所分配的空间开始执行,前2个是数据段,只有从第3条开始是指令代码。
5)程序如下,编写code段中代码,将a段和b段中的数据依次相加,将结果存到C段中。
方法一:两次循环
assume cs:code
a segment
db 1,2,3,4,5,6,7,8
a ends
b segment
db 1,2,3,4,5,6,7,8
b ends
c segment
db 0,0,0,0,0,0,0,0
c ends
code segment
start: mov ax,a
mov ds,ax
mov ax,b
mov es,ax
mov bx,0
mov cx,8
s: mov al,[bx]
add es:[bx],al
inc bx
loop s
mov ax,c
mov ds,ax
mov bx,0
mov cx,8
s0: mov al,es:[bx]
mov [bx],al
inc bx
loop s0
mov ax,4c00h
int 21h
code ends
end start
方法二:一次循环
assume cs:code
a segment
db 1,2,3,4,5,6,7,8
a ends
b segment
db 1,2,3,4,5,6,7,8
b ends
c segment
db 0,0,0,0,0,0,0,0
c ends
code segment
start: mov ax,a
mov ds,ax ;ds指向a段地址
mov ax,b
mov es,ax ;es指向b段地址
mov bx,0
mov cx,8
s: mov al,[bx]
add al,es:[bx]
mov dx,c
mov ds,dx ;ds指向c段地址
mov [bx],al
mov ax,a
mov ds,ax ;重新将ds指向a段(好像此处还能改进)
inc bx
loop s
mov ax,4c00h
int 21h
code ends
end start
方法三:程序二的改进版
assume cs:code
a segment
db 1,2,3,4,5,6,7,8
a ends
b segment
db 1,2,3,4,5,6,7,8
b ends
c segment
db 0,0,0,0,0,0,0,0
c ends
code segment
start:
mov ax,a
mov ds,ax
mov ax,b
mov es,ax
mov ax,c
mov ss,ax
mov bx,0
mov cx,8
s: mov ax,[bx]
mov ss:[bx],ax
mov ax,es:[bx]
add ss:[bx],ax
inc bx
loop s
mov ax,4c00h
int 21h
code ends
end start
(6)程序如下,编写code段中代码,用PUSH指令将A段中的前8个字型数据,逆序存储到B段中。
assume cs:code
a segment
dw 1,2,3,4,5,6,7,8
a ends
b segment
dw 0,0,0,0,0,0,0,0
b ends
code segment
start: mov ax,a
mov ds,ax ;ds指向a段
mov ax,b
mov bx,0 ;ds:bx指向a段的第1个单元
mov ss,ax
mov sp,16 ;设置栈顶指向b:16
mov cx,8
s: push [bx]
add bx,2
loop s ;将a段中0~16个单元逆次入栈
code ends
end start