1、1、 什么是堆栈?他的工作原则是什么?它的基本操作有哪两个?对应哪两种指令? 堆栈是一种按“先进后出”原则存取数据的存储区域。 堆栈的两种基本操作是压栈和出栈,对应的指令是PUSH和POP。 第2章 8086的指令系统 〔习题2.1〕已知DS=2000H、BX=0100H、SI=0002H,存储单元[20100H]~[20103H]依次存放12 34 56 78H,[21200H]~[21203H]依次存放2A 4C B7 65H,说明下列每条指令执行完后AX寄存器的内容。 (1) mov ax,1200h (2) mov ax,bx (3) mov ax,[1200h]
2、4) mov ax,[bx] (5) mov ax,[bx+1100h] (6) mov ax,[bx+si] (7) mov ax,[bx][si+1100h] 〔解答〕 (1)AX=1200H (2)AX=0100H (3)AX=4C2AH ;偏移地址=bx=0100h (4)AX=3412H ;偏移地址=bx=0100h (5)AX=4C2AH ;偏移地址=bx+1100h=1200h (6)AX=7856H ;偏移地址=bx+si=0100h+0002h=0102h (7)AX=65B7H ;偏移地址=bx+si+1100h=0100h
3、0002h+1100h=1202h 〔习题2.2〕指出下列指令的错误 (1) mov cx,dl (2) mov ip,ax (3) mov es,1234h (4) mov es,ds (5) mov al,300 (6) mov [sp],ax (7) mov ax,bx+di (8) mov 20h,ah 〔解答〕 (1)两操作数类型不匹配 (2)IP指令指针禁止用户访问 (3)立即数不允许传给段寄存器 (4)段寄存器之间不允许传送 (5)两操作数类型不匹配 (6)目的操作数应为[ SI ] (7)源操作数应为 [BX+DI] (8)立即数
4、不能作目的操作数 〔习题2.3〕已知数字0 ~ 9对应的格雷码依次为:18H、34H、05H、06H、09H、0AH、0CH、11H、12H、14H,它存在于以table为首地址(设为200H)的连续区域中。请为如下程序段的每条指令加上注释,说明每条指令的功能和执行结果。 lea bx,table mov al,8 xlat 〔解答〕 lea bx,table ;获取table的首地址,BX=200H mov al,8
5、传送欲转换的数字,AL=8 xlat ;转换为格雷码,AL=12H P35 〔习题2.4〕什么是堆栈,它的工作原则是什么,它的基本操作有哪两个,对应哪两种指令? 〔解答〕 堆栈是一种按“先进后出”原则存取数据的存储区域,位于堆栈段中,使用SS段寄存器记录其段地址;它的工作原则是先进后出;堆栈的两种基本操作是压栈和出栈,对应的指令是PUSH和POP。 〔习题2.5〕已知SS = 2200H、SP = 00B0H,画图说明执行下面指令序列时,堆栈区和SP的内容如何变化? mov ax
6、8057h push ax mov ax,0f79h push ax pop bx pop [bx] 〔解答〕 mov ax,8057h push ax mov ax,0f79h push ax pop bx ;bx=0f79h pop [bx] ;DS:[0f79h]=
7、8057h 〔习题2.6〕给出下列各条指令执行后AL值,以及CF、ZF、SF、OF和PF的状态: mov al,89h add al,al add al,9dh cmp al,0bch sub al,al dec al inc al 〔解答〕 mov al,89h ; AL=89h CF ZF SF OF PF add al,al ; AL=12h
8、 1 0 0 1 1 ; 1000 1001 +1000 1001 10001 0010 add al,9dh ; AL=0afh 0 0 1 0 1 ; 0001 0010 + 1001 1101 1010 1111 cmp al,0bch ; AL=0afh 1 0 1 0 1 ; 1010 1111 -1011 1100 * 0100 0011 sub al,al ; AL=00h 0 1 0 0
9、 1 dec al ; AL=0ffh 0 0 1 0 1 ; 0000 0000 - 0000 0001 *1111 1111 inc al ; AL=00h 0 1 0 0 1 ;1111 1111 +0000 0001 *1111 1111 〔习题2.7〕设X、Y、Z均为双字数据,分别存放在地址为X、X+2;Y、Y+2;Z、Z+2的存储单元中,它们的运算结果存入W单元。阅读如下程序段,给出运算公式。 mov
10、ax,X mov dx,X+2 add ax,Y adc dx,Y+2 add ax,24 adc dx,0 sub ax,Z sbb dx,Z+2 mov W,ax mov W+2,dx 〔解答〕 W=X+Y+24-Z 〔习题2.8〕请分别用一条汇编语言指令完成如下功能: (1)把BX寄存器和DX寄存器的内容相加,结果存入D
11、X寄存器。 (2)用寄存器BX和SI的基址变址寻址方式把存储器的一个字节与AL寄存器的内容相加,并把结果送到AL中。 (3)用BX和位移量0B2H的寄存器相对寻址方式把存储器中的一个字和CX寄存器的内容相加,并把结果送回存储器中。 (4)用位移量为0520H的直接寻址方式把存储器中的一个字与数3412H相加,并把结果送回该存储单元中。 (5)把数0A0H与AL寄存器的内容相加,并把结果送回AL中。 〔解答〕 (1)ADD DX,BX (2)ADD AL,[BX+SI] (3)ADD [BX+0B2H],CX (4)ADD WORD PTR [0520H],3412H (5)
12、ADD AL,0A0H 〔习题2.9〕设X、Y、Z、V均为16位带符号数,分别装在X、Y、Z、V存储单元中,阅读如下程序段,得出它的运算公式,并说明运算结果存于何处。 mov ax,X ;ax=X imul Y ;DX.AX=X*Y mov cx,ax ;cx=X*Y的低16位 mox bx,dx ;bx=X*Y的高16位 mov ax,Z ;ax=Z cwd add
13、 cx,ax ;cx=Z的低16位+X*Y的低16位 adc bx,dx ;bx=Z的高16位+X*Y的高16位+低位进位 sub cx,540 ;cx=Z的低16位+X*Y的低16位-540 sbb bx,0 ;bx=Z的高16位+X*Y的高16位+低位进位-低位借位 mov ax,V ;ax=V cwd sub ax,cx ;ax=V的低16位-(Z的低16位+X*Y的低16位-540)
14、 sbb dx,bx ;dx=V的高16位-(Z的高16位+X*Y的高16位+低位进位-低位借位)-低位借位 idiv X ;/X 〔解答〕 [V-(X*Y+Z-540)]/X AX存商,DX存余数 〔习题2.10〕指出下列指令的错误: (1) xchg [si],30h (2) pop cs (3) sub [si],[di] (4) push ah (5) adc ax,ds (6) add [si],80h (7) in al,3fch (8) out dx,ah 〔解答〕
15、 (1)xchg的操作数不能是立即数 (2)不应对CS直接赋值 (3)两个操作数不能都是存储单元 (4)堆栈的操作数不能是字节量 (5)adc的操作数不能是段寄存器 (6)没有确定是字节还是字操作 (7)in不支持超过FFH的直接寻址 (8)out只能以AL/AX为源操作数 〔习题2.11〕给出下列各条指令执行后的结果,以及状态标志CF、OF、SF、ZF、PF的状态。 mov ax,1470h and ax,ax or ax,ax xor ax,ax
16、not ax test ax,0f0f0h 〔解答〕 mov ax,1470h ;AX=1470H CF ZF SF OF PF and ax,ax ;AX=1470H 0 0 0 0 0 ;0001 0100 0111 0
17、000 or ax,ax ;AX=1470H 0 0 0 0 0 xor ax,ax ;AX=0000H 0 1 0 0 1 not ax ;AX=FFFFH 0 1 0 0 1 test ax,0f0f0h ;AX=FFFFH 0 0 1 0 1 注意:MOV和NOT指令不影响标志位;其他逻辑指令使CF
18、=OF=0,根据结果影响其他标志位。 〔习题2.12〕假设例题2.32的程序段中,AX = 08H,BX = 10H,请说明每条指令执行后的结果和各个标志位的状态。 〔解答〕 指令 ; 执行结果 CF OF SF ZF PF mov si,ax ; SI=AX=0008H - - - - - shl si,1 ; SI=2*AX=0010H 0
19、 0 0 0 0 add si,ax ; SI=3*AX=0018H 0 0 0 0 1 mov dx,bx ; DX=BX=0010H 0 0 0 0 1 mov cl,03h ; CL=03H 0 0 0 0 1 shl dx,cl ; DX=8*BX=0080H
20、 0 u 0 0 0 sub dx,bx ; DX=7*BX=0070H 0 0 0 0 0 add dx,si ; DX=7*BX+3*AX=0088H 0 0 0 0 1 注意:逻辑左移N次相当于无符号整数乘以2的N次方,逻辑右移N次相当于无符号整数除以2的N次方。移位指令根据移位的数据设置CF,根据移位后的结果影响SF,ZF,PF。在进行一位移位时,根据最高符号位是否改变设置OF,如改变则OF=1。另外,程序注释用“u”表示
21、标志无定义(不确定),“-”表示无影响。 〔习题2.13〕编写程序段完成如下要求: (1)用位操作指令实现AL(无符号数)乘以10 (2)用逻辑运算指令实现数字0 ~ 9的ASCII码与非压缩BCD码的互相转换 (3)把DX.AX中的双字右移4位 〔解答〕 (1) ;不考虑进位 mov bl,al mov cl,3 shl al,cl ;*8 add al,bl ;shl bl,1 ad
22、d al,bl ;考虑进位 xor ah,ah mov bx,ax mov cl,3 shl ax,cl add ax,bx ;shl bx,1 add ax,bx (2)数字0~9的ASCII码是:30h~39h 非压缩BCD码的0~9是:00h~09h 方法一: and al,0fh ;实现ASCII到非压缩BCD码的转换
23、 or al,30h ;实现非压缩BCD码到ASCII的转换 方法二: xor al,30h ;求反D5D4位,其他不变 ;即高4位为3,则变为0;高4位为0,则变为3 (3) mov cl,4 again: shr dx,1 ;实现逻辑右移 ;采用“sar dx,1”,则实现算术右移 rcr ax,1 dec c
24、l jnz again 〔习题2.14〕已知AL = F7H(表示有符号数-9),分别编写用SAR和IDIV指令实现的除以2的程序段,并说明各自执行后,所得的商是什么? 〔解答〕 (1)用sar编写 mov al,0f7h ; -9送AL 1111 1001 sar al,1 ; 结果:AL=1111 1100B=0FBH 即-5 (2)用idiv编写 mov al,0f7h ; -9送al
25、 cbw ; 字节符号扩展位字 mov bl,2 ; 注意除数不可为立即数 idiv bl ; 结果:商为al=fch (-4) ; 余数:ah=ffh (-1) 结论:符号数的除法用idiv 准确。 〔习题2.15〕已知数据段500h ~600h处存放了一个字符串,说明下列程序段执行后的结果: mov si,600h mov d
26、i,601h mov ax,ds mov es,ax mov cx,256 std rep movsb 〔解答〕 〔习题2.16〕说明下列程序段的功能 cld mov ax,0fefh mov cx,5 mov bx,3000h mov es,bx mov di,2000h rep stosw
27、〔解答〕 将es:di (即3000:2000h或32000h)开始的5个单元的内容置为0fefh。 〔习题2.17〕指令指针IP是通用寄存器还是专用寄存器?有指令能够直接它赋值吗?哪类指令的执行会改变它的值? 〔解答〕 指令指针IP不是通用寄存器,不可直接赋值,属于专用寄存器。有且仅有循环、转移、子程序调用和返回、中断类等指令可以改变它的值。 〔习题2.18〕控制转移类指令中有哪三种寻址方式? 〔解答〕 控制转移类指令的寻址方式:相对寻址、直接寻址方式和间接寻址方式(又可以分成寄存器和存储器间接寻址)。 〔习题2.19〕什么是短转移short jump、
28、近转移near jump和远转移far jump?什么是段内转移和段间转移?8086有哪些指令可以实现段间转移? 〔解答〕 短转移:指段内-128~127之间的转移,位移量用一个字节表示 近转移:指段内±32K之间的转移,位移量用一个字表示 远转移:指段间1MB范围的转移 段内转移:指在同一个代码段内的转移,可以是短转移或者近转移 段间转移:指转移到另外一个代码段,就是远转移 8086/8088CPU的JMP、CALL和INT n指令可以实现段间转移 〔习题2.20〕8086的条件转移指令的转移范围有多大?实际编程时,你如何处理超出范围的条件转移? 〔解答〕 808
29、6的条件转移的转移范围:在当前指令地址的 +127~-128之内。 如条件转移的转移范围超出此范围,可在此范围内安排一条无条件转移,再转移到范围外的目标地址。 〔习题2.21〕假设DS=2000H,BX=1256H,SI=528FH,位移量TABLE=20A1H,[232F7H]=3280H,[264E5H]=2450H,试问执行下列段内间接寻址的转移指令后,转移的有效地址是什么? (1) JMP BX (2) JMP TABLE[BX] (3) JMP [BX][SI] 〔解答〕 (1)转移的有效地址EA= BX=1256H (2)转移的有效地址EA= [DS:20A1
30、H+1256H]=[232F7]=3280H (3)转移的有效地址EA= [DS:1256H+528FH]=264E5H=2450H 〔习题2.22〕判断下列程序段跳转的条件 (1) xor ax,1e1eh je equal (2)test al,10000001b jnz there (3)cmp cx,64h jb there 〔解答〕 (1)AX=1e1eh(异或后为
31、0) (2)AL的D0或D7至少有一位为1 (3)CX(无符号数)< 64h 〔习题2.23〕设置CX = 0,则LOOP指令将循环多少次?例如: mov cx,0 delay: loop delay 〔解答〕 216次。 〔习题2.24〕假设AX和SI存放的是有符号数,DX和DI存放的是无符号数,请用比较指令和条件转移指令实现以下判断: (1)若DX > DI,转到above执行; (2)若AX > SI,转到greater执行; (3)若CX = 0,转到zero执行; (4)若AX-SI产生溢出,转到overflow
32、执行; (5)若SI≤AX,转到less_eq执行; (6)若DI≤DX,转到below_eq执行。 〔解答〕 (1)若DX > DI,转到above执行 cmp dx,di ja above ;=jnbe above (2)若AX > SI,转到greater执行 cmp ax,si jg greater ;=jnle greater (3)若CX = 0,转到zero执行 cmp cx,0
33、 jz zero ;= jcxz zero (4)若AX-SI产生溢出,转到overflow执行; cmp ax,si jo overflow (5)若SI≤AX,转到less_eq执行; cmp si,ax ; cmp ax,si jle less_eq ; jge less_eq (6)若DI≤DX,转到below_eq执行。 cmp di,dx ; cmp dx,
34、di jbe below_eq ; jae below_eq 〔习题2.25〕有一个首地址为array的20个字的数组,说明下列程序段的功能。 mov cx,20 mov ax,0 mov si,ax sum_loop: add ax,array[si] add si,2 loop sum_loop mov total,ax 〔解答〕 将首地址为array得20个字的数组求和,并将结果存入 to
35、tal 单元中。 〔习题2.26〕按照下列要求,编写相应的程序段: (1)起始地址为string的主存单元中存放有一个字符串(长度大于6),把该字符串中的第1个和第6个字符(字节量)传送给DX寄存器。 (2)从主存buffer开始的4个字节中保存了4个非压缩BCD码,现按低(高)地址对低(高)位的原则,将它们合并到DX中。 (3)编写一个程序段,在DX高4位全为0时,使AX = 0;否则使AX = -1。 (4)有两个64位数值,按“小端方式”存放在两个缓冲区buffer1和buffer2中,编写程序段完成buffer1-buffer2功能。 (5)假设从B800h : 0开
36、始存放有100个16位无符号数,编程求它们的和,并把32位的和保存在DX.AX中。 (6)已知字符串string包含有32KB内容,将其中的’$’符号替换成空格。 (7)有一个100个字节元素的数组,其首地址为array,将每个元素减1(不考虑溢出)存于原处。 (8)统计以 ’$’ 结尾的字符串srting的字符个数。 〔解答〕 (1)解答: mov si,0 mov dl,string[si] ;第1个字符送dl寄存器:mov dl,stirng[0] mov si,5 mov dh
37、string[si] ;第6个字符送dh寄存器:mov dl,stirng[5] (2)解答: xor si,si ;si清零 mov al,buffer[si] ;第一字节 inc si mov ah,buffer[si] ;第二字节 mov cl,4 shl ah,cl ;BCD码移到高半字节 or al,ah ;组合成压缩BCD码
38、 mov dl,al ;存入dl寄.. inc si mov al,buffer[si] ;第三字节 inc si mov ah,buffer[si] ;第四字节 mov cl,4 shl ah,cl ;BCD码移到高半字节 or al,ah ;组合成压缩BCD码 mov dh,al
39、 ;存入dh寄.. (3)解答: test dx,0f000h ; test dh,0f0h jz next ; jnz next mov ax,-1 ; mov ax,0 jmp again next: mov ax,0 ; mov ax,0ffffh again: ... (4)解答: mov ax, word ptr buffer1
40、 sub ax, word ptr buffer2 ;先减低16位 mov dx, word ptr buffer1+2 sbb dx, word ptr buffer2+2 ;后减高16位,需减低16位的借位 (5)解答: mov ax,0b800h mov ds,ax ;段地址 xor si,si ;地址偏移量si=0 xor dx,dx ;和的高字dx=0
41、 mov cx,99 ;加的次数 mov ax,[si] ;第一个数 again: inc si ;指向下一个字单元 inc si add ax,[si] ;加下一个数 jnc noc ;无进位转 inc dx ;有进位dx=dx+1 noc: dec cx ;次数-1 jnz cx,again ;非0继续
42、加 (6)解答1:不使用串操作指令(更好) mov si,offset string mov cx,8000h ; 32k=2^15=8000h again: cmp byte ptr [si], ‘$’ ;‘$’ =24h jnz next ; 不要采用 jz 进行分支 mov byte ptr [si], ‘ ’ ;‘ ’=20h next: inc si loop
43、again ; dec cx ; jnz again (6)解答2:使用串操作指令 mov di,offset string mov al,’$’ mov cx,8000h cld again: scasb jnz next mov byte ptr es : [di-1], ‘ ’ next: loop again
44、 (7)解答1: mov si,offset array mov cx,100 again: dec byte ptr [si] inc si loop again (7)解答2: xor si,si ; si<--0 mov cx,100 ; 循环次数 again: dec array[si] inc si loop
45、again (7)解答3: mov si,offset array mov di,si mov ax,ds mov es,ax mov cx,100 cld again: lodsb dec al stosb loop again (8)解答: xor si,si ;si<--0 coun:
46、 cmp string[si],'$' je done inc si jmp coun done: ... 〔习题2.27〕对下面要求,分别给出3种方法,每种方法只用一条指令。 (1)使CF=0 (2)使AX=0 (3)同时使AX=0和CF=0 〔解答〕 (1)解答: clc and ax,ax or ax,ax (2)解答: xor
47、 ax,ax and ax,0 mov ax,0 (3)解答: and ax,0 xor ax,ax sub ax,ax 〔习题2.28〕参照本习题的示意图,分析调用序列,画出每次调用及返回时的堆栈状态。其中CALL前是该指令所在的逻辑地址;另外,段内直接调用指令的机器代码的字节数为3,段间直接调用指令则为5个字节。 〔解答〕 主程序转子suba时段内调用:断点1为2000h:0400h+3, 转子是只将IP压栈。 suba转子subb时段间调用:断点2为2000h:0840h
48、5, 转子时须将cs段地址和IP压栈 suba转子subc时段内调用:断点3为2000h:0c021h+3,转子是只将IP压栈。 注:压栈时先修改sp再压入断点,弹栈时先弹出断点再修改sp。 〔习题2.29〕已知AX 、BX存放的是4位压缩BCD表示的十进制数,请说明如下子程序的功能和出口参数。 add al,bl daa xchg al,ah adc al,bh daa xchg al,ah ret 〔解答〕
49、 压缩BCD码加法:AX←AX+BX 出口参数:AX=BCD码和 〔习题2.30〕AAD指令是用于除法指令之前,进行非压缩BCD码调整的。实际上,处理器的调整过程是:AL←AH×10+AL,AH←0。如果指令系统没有AAD指令,请用一个子程序完成这个调整工作。 〔解答〕 shl ah,1 ;ah=2*a (设原ah=a) mov dl,ah ;dl=2*a mov cl,2 ;设定移位次数 shl ah,cl
50、 ;ah=8*a add ah,dl ;ah=10*a add al,ah ;al=10*a+al xor ah,ah ;清零ah int 3 ;返回DOS 注意: 入口:AX中存放有“和”(两非压缩BCD码) 出口:AL中 已为调整后的二进制数 〔习题2.31〕解释如下有关中断的概念: (1) 内部中断和外部中断 (2) 单步中断和断点中断 (3) 除






