首页 > 代码库 > 《汇编语言》总结05 —— 数据处理的两个基本问题

《汇编语言》总结05 —— 数据处理的两个基本问题

(一)前述

标题所说的两个基本的问题为:

  1. 处理的数据在什么地方

  2. 要处理的数据有多长

这两个问题,在机器指令中必须给以明确或隐含的说明,否则计算机无法工作。

我们定义的描述性符号: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个概念来表达数据的位置

  1. 立即数(idata)

    mov ax,1

    add bx,2000h

    or bx,00010000B

    mov al,‘a‘

  2. 寄存器:指令要处理的数据在寄存器中

    mov ax,bx

    mov ds,ax

    push bx

    mov ds:[0],bx

    push ds

    mov ss,ax

    mov sp,ax

  3. 段地址和偏移地址

    存放段地址的寄存器可以是默认的,如下:

    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。所以在机器指令中要指明,指令进行的是字操作还是字节操作。

  1. 通过寄存器名指明要处理的数据的尺寸

    下面指明是字操作

    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

  2. 在没有寄存器名存在的情况下,用操作符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

  3. 有些指令默认了访问的是字单元还是字节单元,比如,push [1000H]就不用指明访问的是字单元还是字节单元,因为push指令只进行字操作

(六)div指令

div是除法指令,使用这个指令需要注意以下几个问题:

  1. 除数:有8位和16位两种,在一个reg或内存单元中

  2. 被除数:默认放在axdx和ax中,如果除数为8位,被除数则为16位,默认在ax中存放;如果除数为16位,被除数则为32位,在dx和ax中存放,dx存放高16位,ax存放低16位

  3. 结果:如果除数为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

《汇编语言》总结05 —— 数据处理的两个基本问题