数据处理的两个基本问题
1.bx、si、di、bp
1.1 可以用来寻址的寄存器
在8086CPU中,只有bx、bp、si、di这四个寄存器可以用在[]中来进行内存单元的寻址。
正确的指令:
mov ax,[bx]
mov ax,[bx+si]
mov ax,[bx+di]
mov ax,[bp]
mov ax,[bp+si]
mov ax,[bp+di]
错误的指令:
mov ax,[cx]
mov ax,[ax]
mov ax,[dx]
mov ax,[ds]
1.2 四个寄存器的组合使用
在”[…]”中,这4个寄存器(bx、bp、si、di)可以单个出现,或只能以四种组合出现:
bx与si、bx与di、bp与si、bp与di
错误的用法:
mov ax,[bx+bp]
mov ax,[di+si]
1.3 寄存器bp
只要在[…]中使用寄存器bp,而指令中没有显性地给出段地址,段地址就默认在ss中。比如:
mov ax,[bp]
含义为:(ax) = ((ss)*16 + (bp))
mov ax,[bp+idata]
含义为:(ax) = ((ss)*16 + (bp) + idata)
2.机器指令处理的数据所在位置
绝大部分机器指令都是进行数据处理的指令,处理大致可分为三类:读取、写入、运算。
在机器指令这一层来讲,并不关心数据的值是多少,而关心指令执行前一刻,它将要处理的数据所在的位置。
指令在执行前,所要处理的数据可以在三个地方:CPU內部、内存、端口
例:
3.汇编语言中数据位置的表达
汇编语言中用三个概念来表达数据的位置:
- 立即数(idata)
- 寄存器
- 段地址(SA)和偏移地址(EA)
3.1 立即数(idata)
对于直接包含在机器指令中的数据(执行前在CPU的指令缓冲器中),在汇编语言中称为:立即数(idata),在汇编指令中直接给出。例如:
mov ax,1
对应机器码:B80100
执行后的结果:(ax) = 1
3.2 寄存器
指令要处理的数据在寄存器中,在汇编指令中给出相应的寄存器名,例如:
mov ax,bx
对应的机器码:89D8
执行结果:(ax) = (bx)
3.3 段地址(SA)和偏移地址(EA)
指令要处理的数据在内存中,在汇编指令中可用[x]的格式给出EA,SA在某个段寄存器中。
- 存放段地址的寄存器可以是默认的
- 存放段地址的寄存器也可以显性地给出
示例:
默认段地址为ds
mov ax,[0]
;偏移地址为0可以直接写[0]
mvo ax,ds:[9]
mov ax,[bx+8]
mvo ax,[bx+si]
默认段地址为ss
mov ax,[bp]
mov ax,[bp+8]
mov ax,[bp+si]
显性地给出存放段地址的寄存器
mov ax,ds:[bp]
含义:(ax) = ((ds)*16 + (bp))
mov ax,es:[bx]
含义:(ax) = ((es)*16 + (bx))
mov ax,ss:[bx+si]
含义:(ax) = ((ss)*16 + (bx) + (si))
4.寻址方式
当数据存放在内存中的时候,我们可以用多种方式来给定这个内存单元的偏移地址,这种定位内存单元的方法一般被称为寻址方式。
5.指令要处理的数据有多长
8086CPU的指令,可以处理两种尺寸的数据,byte和word。所以在机器指令中要指明,指令进行的是字操作还是字节操作。
5.1 通过寄存器名指明要处理的数据的尺寸
下面的指令中,寄存器指明了指令进行的是字操作:
mov ax,1
mov bx,ds:[0]
mov ds,ax
inc ax
5.2 用操作符X ptr指明内存单元的长度
在没有寄存器名存在的情况下,用操作符X ptr指明内存单元的长度,X在汇编指令中可以为word和byte。
下面的指令中,用word ptr指明了指令访问的内存单元是一个字单元:
mov word ptr ds:[0],1
inc word ptr [bx]
在没有寄存器参与的内存单元访问指令中,用word ptr或byte ptr显性地指明所要访问的内存单元的长度是很有必要的。
例如:
假设我们用Debug查看内存的结果如下:
2000:1000 FF FF FF FF FF FF ……
那么指令:
1 | mov ax,2000H |
将使内存中的内容变为:
2000:1000 01 FF FF FF FF FF ……
而指令
1 | mov ax,2000H |
将使内存中的内容变为:
2000:1000 01 00 FF FF FF FF ……
5.3 其他方法
有些指令默认了访问的是字单元还是字节单元。
比如:push [1000H]就不用指明访问的是字单元还是字节单元。因为push指令只进行字操作。