(一)前述
标题所说的两个基本的问题为:
处理的数据在什么地方?
要处理的数据有多长?
这两个问题,在机器指令中必须给以明确或隐含的说明,否则计算机无法工作。
我们定义的描述性符号:reg和sreg。
reg表示一个寄存器,用sreg表示一个段寄存器。
reg的集合包括:ax、bx、cx、dx、ah、al、bh、bl、ch、cl、dh、dl、sp、bp、si、di;
sreg的集合包括:ds、ss、cs、es。
(二)bx、si、di和bp
在8086CPU中,只有这4个寄存器可以用在“[...]”中来进行内存单元的寻址。下面的指令是正确的:
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]
在[...]中,这4个寄存器可以单个出现,或只能以4种组合出现:bx和si、bx和di、bp和si、bp和di。
下面的指令是错误的:
mov ax,[bx+bp]
mov ax,[si+di]
只要在[...]中使用寄存器bp,而指令中没有显性地给出段地址,段地址就默认在ss中,如下:
mov ax,[bp]
mov ax,[bp+idata]
mov ax,[bp+si]
mov ax,[bp+si+idata]
(三)机器指令处理的数据在什么地方
处理的数据可以在3个地方:CPU内部、内存、端口。
(四)汇编语言中数据位置的表达
可以用3个概念来表达数据的位置
立即数(idata)
mov ax,1
add bx,2000h
or bx,00010000B
mov al,‘a‘
寄存器:指令要处理的数据在寄存器中
mov ax,bx
mov ds,ax
push bx
mov ds:[0],bx
push ds
mov ss,ax
mov sp,ax
段地址和偏移地址
存放段地址的寄存器可以是默认的,如下:
mov ax,[0]
mov ax,[di]
mov ax,[bx+8]
mov ax,[bx+si]
mov ax,[bx+si+8]
等指令,段地址默认在ds中;
mov ax,[bp]
mov ax,[bp+8]
mov ax,[bp+si]
mov ax,[bp+si+8]
等指令,段地址默认在ss中。存放段地址的寄存器也可以显性给出,如下:
mov ax,ds:[bp]
mov ax,es:[bx]
mov ax,ss:[bx+si]
mov ax,cs:[bx+si+8]
(五)指令要处理的数据有多长
8086CPU的指令,可以处理两种尺寸的数据,byte和word。所以在机器指令中要指明,指令进行的是字操作还是字节操作。
通过寄存器名指明要处理的数据的尺寸
下面指明是字操作
mov ax,1
mov bx,ds:[0]
mov ds,ax
mov ds:[0],ax
inc ax
add ax,1000
下面的指令中,寄存器指明了指令进行的是字节操作
mov al,1
mov al,bl
mov al,ds:[0]
mov ds:[0],al
inc al
add al,100
在没有寄存器名存在的情况下,用操作符X ptr指明内存单元的长度,X在汇编指令中可以为word或byte。
下面的指令中,用word ptr指明了指令访问的内存单元是一个字单元
mov word ptr ds:[0],1
inc word ptr [bx]
inc word ptr ds:[0]
add word ptr [bx],2
下面的指令中,用byte ptr指明了指令访问的内存单元是一个字节单元
mov byte ptr ds:[0],1
inc byte ptr [bx]
inc byte ptr ds:[0]
add byte ptr [bx],2
有些指令默认了访问的是字单元还是字节单元,比如,push [1000H]就不用指明访问的是字单元还是字节单元,因为push指令只进行字操作。
(六)div指令
div是除法指令,使用这个指令需要注意以下几个问题:
除数:有8位和16位两种,在一个reg或内存单元中
被除数:默认放在ax或dx和ax中,如果除数为8位,被除数则为16位,默认在ax中存放;如果除数为16位,被除数则为32位,在dx和ax中存放,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)的余数
(七)伪指令dd
前面我们用db和dw定义字节型数据和字型数据,dd是用来定义dword(double word)型数据的,比如:
data segment
db 1
dw 1
dd 1
data ends
在data段中定义了3个数据
第一个数据为01H,在data:0处,占1个字节;
第二个数据为0001H,在data:1处,点1个字;
第三个数据为00000001H,在data:3处,占2个字;
(八)dup指令
dup是一个操作符,它是编译器识别处理的符号,用来进行数据的重复,比如:
db 3 dup (0)
定义了3个字节,它们的值都是0,相当于db 0,0,0。
db 3 dup (0,1,2)
定义了9个字节,它们是0、1、2、0、1、2、0、1、2,相当于db 0,1,2,0,1,2,0,1,2。
可见,dup的使用格式如下:
db 重复的次数 dup (重复的字节型数据)
dw 重复的次数 dup (重复的字型数据)
dd 重复的次数 dup (重复的双字型数据)
stack segment
dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
dw 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0
stack ends
相当于如下:
stack segment
db 200 dup (0)
stack ends
总结完毕!
本文出自 “凡凡不会玩” 博客,请务必保留此出处http://liaofan.blog.51cto.com/12295212/1918764
原文地址:http://liaofan.blog.51cto.com/12295212/1918764