mov/add指令
mov a,b 将b中的数据移动到a中。
指令中数据从哪送到哪一定要指明,寄存器用寄存器名来指明,内存单元用内存单元的地址来指明。[…]表示一个内存单元,[0]中的0表示内存单元的偏移地址。只有偏移地址是不能定位一个内存单元的,指令执行时,8086CPU自动取ds中的数据作为内存单元的段地址。
例1:将10000H(用段地址和偏移地址表示为1000:0)中的数据读到al中
mov bx,1000H ;将10000H中的数据送入bx中
mov ds,bx ;将bx中的数据送入ds中
mov al,[0] ;偏移为0,相当于不动,把当前位置的数据送入al中
mov指令可以将数据从内存单元中移到寄存器,也可以从寄存器中移到内存单元,但是8086CPU不支持将数据直接送入段寄存器,ds是一个段寄存器,所以指令mov ds 10000H是非法的。那么怎么把10000H数据送入ds呢?那就只好用一个寄存器来进行中转,先把数据10000H送入一个一般的寄存器如bx,然后再把bx中的内容送入ds。
例2:将123B0H~123B9H的内存单元定义为数据段,累加这个数据段的前三个单元的数据。
mov ax,123B0H
mov ds,ax ;将123B0H送入ds中,作为数据段的段地址
mov al,0 ;用al存放累加结果
add al,[0] ;将数据段第一个单元(偏移地址为0)的数据加到al中
add al,[1] ;将数据段第二个单元(偏移地址为1)的数据加到al中
add al,[2] ;将数据段第三个单元(偏移地址为2)的数据加到al中
例3:写几条指令,累加数据段的前3个字型数据。
字单元,存放一个字型数据的(16位)内存单元,由两个地址连续的内存单元组成。高地址内存单元存放字型数据的高位字节,低地址内存单元存放字型数据的低位字节。
mov ax,123B0H
mov ds,ax ;将123B0H送入ds中,作为数据段的段地址
mov al,0 ;用al存放累加结果
add al,[0] ;将数据段第一个字(偏移地址为0)加到al中
add al,[2] ;将数据段第二个字(偏移地址为2)加到al中
add al,[4] ;将数据段第三个字(偏移地址为4)加到al中
Push/Pop指令
栈机制,类似于往盒子里放书,后进先出的特点。push ax表示把寄存器里的数据送入栈中,pop ax表示从栈顶取出数据送入ax。这些操作都是以字为单位进行的。push ax入栈指令执行时,要把寄存器中的数据放到当前栈顶单元的上方,成为新的栈顶元素,pop指令执行时,要从栈顶取出数据,然后送入寄存器中。
那么CPU怎么知道当前要执行的指令的位置呢?8086CPU中有两个寄存器,段寄存器SS和寄存器SP,栈顶的地址放在SS中,偏移地址放在SP中。任何时刻,SS:SP指向栈顶元素,CPU从SS和SP中得到栈顶的地址。
下面是入栈、出栈的执行过程,注意,字型数据用2个单元存放,
mov ax,0123H
push ax
mov bx,2266H
push bx
mov cx,1122H
push cx
pop ax
pop bx
pop cx

出栈后,SS:SP指向新的栈顶1000EH,pop操作前的1000CH处的2266依然存在,只是已不在栈中,当再次执行push指令,SS:SP移到1000CH,并向栈中写入新的数据时,它将被覆盖。
例4:编程,将10000H~1000FH这段空间当作栈,初始状态栈为空,将ax,bx,ds的数据入栈。
mov ax,1000H
mov ss,ax ;设置段的栈地址,用ax寄存器中转
mov sp 0010H ;设置栈的偏移地址
push ax
push bx
push ds
例5:编程,将10000H~1000FH这段空间当作栈,初始状态栈为空,设置AX=001AH,BX=001BH,将ax,bx的数据入栈,将AX,BX清零,从栈中恢复AX,BX原来的内容。
mov ax,1000H
mov ss,ax
mov sp,0010H ;初始化栈顶
mov ax,001AH
mov bx,001BH
push ax
push bx
sub ax,ax ;将ax清零,也可以用mov ax,0,但mov指令机器码是3个字节
sub bx,bx ;sub指令机器码是2个字节
pop bx ;从栈中恢复AX,BX原来的内容
pop ax ;AX,BX中原来的数据是001AH,001BH

例6:编程,将10000H~1000FH这段空间当作栈,初始状态栈为空,设置AX=001AH,BX=001BH,利用栈,交换AX,BX中的数据。
mov ax,1000H
mov ss,ax
mov sp,0010H ;初始化栈顶
mov ax,001AH
mov bx,001BH
push ax
push bx
pop ax ;当前栈顶的数据是bx原来的数据001BH,执行pop指令后,ax=001BH
pop bx ;执行pop ax之后,栈顶的数据是ax原来的数据001AH,所以bx=001AH
例7:在10000H处写入数据2266H,有下面两种方法。
法一:
mov ax,1000H
mov ds,ax
mov ax,2266H
mov [0],ax
法二:(不使用mov 内存单元 寄存器 这类指令)
mov ax,1000H
mov ss,ax
mov sp,2 ;此处应为加2,入栈操作后sp-2,出栈操作后sp+2
mov ax,2266H
push ax
push指令,是在栈顶之上压入新的数据,注意,它的执行过程是,先将记录栈顶偏移地址的SP寄存器中的内容减2,使得SS:SP指向新的内存单元,然后再将寄存器中的数据送入新的内存单元。
所以,要在执行push指令之前,使SS:SP指向10002H(可设SS=1000H,SP=0002H),这样,在执行push ax时,CPU先将SP=SP-2,使得SS:SP指向10000H,再把寄存器ax中的数据送入SS:SP指向的内存单元中,即10000H处。
从上例的分析中可以看出,push和pop指令实际上就是一种内存传送指令,可以在内存和寄存器之间传输数据,与mov指令不同的是,push和pop指令访问的内存单元的地址不是在指令中给出的,而是由SS:SP指出的,同时,push和pop指令还要改变SP中的内容。而且,CPU执行mov只有一步操作,就是传送数据,但执行push和pop需要两步,执行push时,先改变SP,后向SS:SP处传送;执行pop时,先读取SS:SP处的数据,再改变SP。
注意,push和pop等栈操作指令,只会修改SP,也就是说,栈顶最大的变化范围是0~FFFFH。
本文介绍了8086汇编中的mov和add指令,强调了数据传送的方向以及段地址和偏移地址的概念。接着详细讲解了Push和Pop指令,描述了栈的工作原理和这两个指令如何改变栈顶地址并进行数据的入栈和出栈操作。通过多个例子展示了如何在实际编程中应用这些指令。
433

被折叠的 条评论
为什么被折叠?



