栈
- 栈:是一种具有特殊的访问方式的存储空间(后进先出:
Last In Out First
,LIFO
)
8086
会将CS
作为代码段的段地址,将CS:IP
指向的指令作为下一条需要取出执行的指令。8086
会将DS
作为数据段的段地址,mov ax,[address]
就是取出DS:address
的内存数据放到ax寄存器中农工8086会将SS作为栈段的段地址,任意时刻,
SS:SP
指向栈顶元素8086
提供PUSH(入栈)
和POP(出栈)
指令来操作栈段的数据。
比如:push ax
就是将ax
的数据入栈,pop ax
就是将站定的数据送入ax
push ax
push ax
的执行由以下两个步骤完成:
SP = SP- 2, SS:SP
指向当前栈顶前面的单元,以当前栈顶前面的单元为新的栈顶将
ax
中的内容送入SS:SP
指向的内存单元,SS:SP
指向新的栈顶。
pop ax
pop ax
的执行过程和push ax
刚好相反,由以下两步完成
将
SS:SP
指向的内存单元处的数据送入ax
中SP = SP + 2, SS:SP
指向当前栈顶下面的单元,以当前栈顶下面的单元为新的栈顶
思考
如果将
10000H
到1000FH
这段空间当做栈,初始状态栈是空的,此时,SS = 1000H
,SP
应该等于多少?
- 因为初始栈是空的,所以
SP
应该指向栈控件最高地址的下一个单元
- 栈空,
SS:SP
执行栈控件最高地址单元的下一个单元
栈顶超界:
push
pop
push&pop汇编
注意:在8086
中,push、pop
操作的数据都是2
个字节的
练习
编程:
(1)将10000H
到1000FH
这段空间当做栈,初始状态栈是空的;
(2)设置AX=001AH
,BX=001BH
;
(3)利用栈,交换AX
和BX
中的数据.
解:
mov ax, 1000H
mov ss, ax
mov sp, 0010H
mov ax, 001AH
mov bx, 001BH
push ax
push bx
pop ax
pop bx
栈段
对于
8086
来说,在编程时,我们可以根据需要,将一组内存单元定义为一个段我们可以将一组长度为
N(N <= 64KB)
、地址连续、起始地址为16
倍数的内存单元,当做栈空间来使用,称为栈段,比如用10010H ~ 1001FH
这段内存空间来当做栈使用,我们就可以任务10010H ~ 1001FH
是一个栈段,它的段地址为1001H
,长度为16
字节。-
如何使用
push、pop
等栈操作指令访问我们定义的段栈?用SS存放栈段的段地址,用SP�存放栈顶的偏移地址。
段总结
我们可以用一个段存放数据,将它定义为“数据段”
对于数据段,将它的段地址放在DS
中,用mov、 add、sub
等访问内存单元的指令时,CPU
就将我们定义的数据段中的内容当做数据来访问。我们可以用一个段存放代码,将它定义为“代码段”
对于代码段,将它的段地址放在CS
中,将段中第一条指令的偏移地址放在IP
中,这样CPU
就将执行我们定义的代码段中的指令。我们可以用一个段当做栈,将它定义为“栈段”
对于栈段,将它的段地址放在SS
中,将栈顶单元的偏移地址放在SP
中,这样CPU
在需要进行栈操作的时候,比如执行push、pop
指令等,就将我们定义的栈段当做栈空间来用。
练习:
- 解:
将内存地址20000H ~ 2000FH
当做一个栈段,一开始栈段为空,然后将内存地址10000H ~ 1000FH
的数据,顺序入栈到栈段里面。
- 解:
- 将
20000H
,定义为数据端 - 将内存地址
10000H ~ 1000FH
当做一个栈段,并且这个栈是满的,然后将这个栈的数据从栈底pop
出来,放入到数据段。