数据处理的两个基本问题:
- 处理的数据在什么地方?
- 要处理的数据有多长?
为了描述上的简洁,使用两个描述性的符号:
reg表示寄存器:ax,bx,cx,dx,(ah,al,bh...),sp,bp,si,di
sreg表示段寄存器:ds,ss,cs,es
bx、si、di和bp
在8086CPU中,只有这四个寄存器可以用在[...]中来进行内存单元的寻址:
正确:
mov ax,[bx]
mov ax,[bx+si]
mov ax,[bp+di]
错误:
mov ax,[cx]
mov ax,[ds]在[...]中,这四个寄存器可以单个出现,或只能已4种组合出现:bx和si、bx和di、bp和si、bp和di
正确:
mov ax,[bx+si]
mov ax,[bx+di]
mov ax,[bx+si+idata]
mov ax,[bp+di+idata]
错误:
mov ax,[bx+bp]
mov ax,[si+di]只要在[...]中使用寄存器bp,而指令中没有显性地给出段地址,段地址就默认在ss中。
数据在什么地方
在机器指令这一层来讲,并不关心数据的值是多少,而关心指令执行前一刻,它将要处理的数据所在的位置。
- 所要处理的数据可以在3个地方:
CPU内部
内存
端口 - 在汇编语言中数据位置的表达:
立即数(idata)
直接包含在机器指令中的数据(执行前再CPU的指令缓冲器中)
mov ax,1
add bx,2000h
or bx,00010000h
mov al,'a'
寄存器
指令要处理的数据在寄存器中,在汇编指令中给出相应的寄存器名
mov ax,bx
push bx
mov ds:[0],bx
mov ss,ax
段地址和偏移地址
指令要处理的数据在内存中,在汇编指令中可用[X]的格式给出,段地址在某个寄存器中
mov ax,[0]
mov ax,[di]
mov ax,[bp+si+8]
mov ax,ds:[bp]
寻址方式
寻址方式----含义----名称
[idta]--------------EA = idata;SA = (ds)----------------直接寻址
[bx]----------------EA = (bx);SA = (ds)-----------------寄存器间接寻址
[di]-----------------EA = (di);SA = (ds)-----------------寄存器间接寻址
[si+idata]----------EA = (si);SA = (ds)------------------寄存器相对寻址
[bp+idata]---------EA = (bp);SA = (ds)-----------------寄存器相对寻址
[bx+si]-------------EA = (bx)+(si);SA = (ds)------------基址变址寻址
[bp+di]------------EA = (bp)+(di);SA = (ds)------------基址变址寻址
[bx+di+idata]-----EA = (bx)+(di)+idata;SA = (ds)----相对基址变址寻址
[bp+si+idata]-----EA = (bp)+(si)+idata;SA = (ds)----相对基址变址寻址
指令要处理的数据有多长
通过寄存器指明要处理数据的尺寸
- 寄存器指明了进行的是字操作
mov ax,1
mov bx,ds:[0] - 寄存器指明了进行的是字节操作
mov al,1
mov al,ds:[0]
在没有寄存器名存在的情况下,用操作符X ptr指明内存单元的长度,X在汇编指令中可以为word或byte
mov word ptr ds:[0],1 ;指明了指令访问的内存单元是一个字单元
mov byte ptr ds:[0],1 ;指明了指令访问的内存单元是一个字节单元
其他方法
有些指令默认了访问的是字单元还是字节单元,比如push [1000H]只进行字操作。
div指令
div是除法指令:
- 除数:有8位和16位两种。
- 被除数:默认放在AX或DX和AX中,如果除数是8位,被除数则为16位,默认在AX中存放;如果除数16位,则被除数为32位,在DX(高16位)和AX(低16位)中存放.
- 结果:如除数为8位,则AL存储商,AH存储余数;如除数为16位,则AX存储商,DX存储余数。
格式:
div reg
div 内存单元
div byte ptr ds:[0]
含义:
(al) = (ax)/((ds)16+0)的商
(ah) = (ax)/((ds)16+0)的余数
div word ptr es:[0]
含义:
(ax) = [(dx)10000H+(ax)]/((es)16+0)的商
(dx) = [(dx)10000H+(ax)]/((es)16+0)的余数
例子:
利用除法指令计算1001/100
mov ax,1001
mov bl,100
div bl
程序执行后,(al) = 0AH(即10),(ah) = 1(余数为1)
dup
dup是一个操作符,由编译器识别处理。与db、dw、dd等数据定义伪指令配合使用的,用来进行数据的重复。
比如:
db 3 dup (0) ;定义了3个字节,他们的值都是0,相当于db 0,0,0
db 3 dup (0,1,2) ;定义了9个字节,相当于db 0,1,2,0,1,2,0,1,2
db 3 dup ('abc','ABC') ;定义了18个字节,相当于db 'abcABCabcABCabcABC'
dup的使用格式:
db 重复次数 dup (重复的字节型数据)
dw 重复次数 dup (重复的字型数据)
dd 重复次数 dup (重复的双字型数据)