栈
- 栈:是一种具有特殊的访问方式的存储空间(后进先出:
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出来,放入到数据段。
