从入门到精通汇编语言 第四章(内存寻址方式)

参考教程:通俗易懂的汇编语言(王爽老师的书)_哔哩哔哩_bilibili

一、[bx+idata]方式寻址

1、[bx+idata]的含义

(1)[bx+idata]表示一个内存单元,它的偏移地址为(bx)+idata,也即BX寄存器中的数值加上idata。(这为高级语言实现数组提供了便利机制)

(2)举例:“mov ax, [bx+200]”及“mov ax, [200+bx]”的含义。

①将一个内存单元的内容送入AX。

②[bx+200]这个内存单元的长度为2字节(字单元),存放一个字。

③内存单元的段地址在DS中,偏移地址为200加上BX中的数值。

④数学化的描述为:(ax) = ((ds) * 16 + 200 + (bx))。

(3)“[bx+idata]”的其它几种写法:

①[idata+bx],如“mov ax, [idata+bx]”。

②idata[bx],如“mov ax, idata[bx]”。

③[bx].idata,如“mov ax, [bx].idata”。

2、[bx+idata]方式寻址使用举例

(1)目标:对数据段定义的两个长度相同的字符串,将第一个字符串中的小写字母转换为大写字母,将第二个字符串中的大写字母转换为小写字母。

(2)汇编程序:

assume cs:codesg, ds:datasg

datasg segment

        db ‘BaSiC’    ;字符串1

        db ‘mInIx’    ;字符串2

datasg ends

codesg segment

start:  mov ax, datasg

        mov ds, ax   ;初始化段寄存器DS,它要存放数据段的段地址

        mov bx, 0   ;存放数据段偏移地址

        mov cx, 5

    s:  mov al, [bx]   ;将字符串1中的字符依次送入al,每轮循环送入一个

        and al 11011111b  ;将al中的第6位二进制数置0

        mov [bx], al   ;将大写转换后的字符送回它原来的位置

        mov al, [5+bx]   ;将字符串2中的字符依次送入al,每轮循环送入一个

        or al 00100000b  ;将al中的第6位二进制数置1

        mov [5+bx], al   ;将小写转换后的字符送回它原来的位置

        inc bx    ;(bx) = (bx) + 1

        loop s

        mov ax, 4c00h

        int 21h

codesg ends

end start

二、变址寄存器SI和DI

1、SI和DI介绍

(1)SI和DI是8086CPU中和BX功能相近的寄存器,区别是SI和DI不能够分成两个8位寄存器来使用,而BX可以

①BX:通用寄存器,在计算存储器地址时,常作为基址寄存器用。

②SI:源变址寄存器。

③DI:目标变址寄存器。

(2)举例:下列3组指令执行的功能是一样的。

  mov bx, 0  mov ax, [bx+123]

  mov si, 0   mov ax, [si+123]

  mov di, 0   mov ax, [di+123]

2、SI和DI的应用举例

(1)目标:用寄存器SI和DI实现将字符串’welcome to masm!’复制到它后面的数据区中。

(2)汇编程序:

assume cs:codesg, ds:datasg

datasg segment

        db ‘welcome to masm!’

        db ‘................’   ;为复制版字符串预留位置

datasg ends

codesg segment

start:  mov ax, datasg

        mov ds, ax  ;初始化段寄存器DS,它要存放数据段的段地址

        mov si, 0   ;ds:si指向原始字符串首地址

        mov di, 16  ;ds:si指向目的空间首地址

        mov cx, 8   ;一次复制两个字节(两个字符),循环8次即可

    s:  mov ax, [si]  ;将源字符串中的两个字符送入ax,每轮循环送入两个

        mov [di], ax  ;将ax中的两个字符送到目的空间,每轮循环送出两个

        add si, 2   ;(si) = (si) + 2   复制完前面两个字符,指向后面两个字符

        add di, 2   ;(di) = (di) + 2   字符填入空间后,指向后面仍为空的空间

        loop s

        mov ax, 4c00h

        int 21h

codesg ends

end start

三、[bx+si]和[bx+di]方式寻址及[bx+si+idata]和[bx+di+idata]方式寻址

1、[bx+si]和[bx+di]的含义

(1)[bx+si]表示一个内存单元,它的偏移地址为(bx)+(si),也即BX寄存器中的数值加上SI寄存器中的数值[bx+di]表示一个内存单元,它的偏移地址为(bx)+(di),也即BX寄存器中的数值加上DI寄存器中的数值

(2)举例:“mov ax, [bx+si]”的含义。

①将一个内存单元的内容送入AX。

②[bx+si]这个内存单元的长度为2字节(字单元),存放一个字,内存单元的段地址在DS中,偏移地址为BX中的数值加上SI中的数值。

③数学化的描述为:(ax) = ((ds) * 16 + (bx) + (si))。

(3)举例:“mov [bx+di], ax”的含义。

①将一个内存单元的内容送入[bx+di]。

②[bx+di]这个内存单元的长度为2字节(字单元),存放一个字,内存单元的段地址在DS中,偏移地址为BX中的数值加上DI中的数值。

③数学化的描述为:((ds) * 16 + (bx) + (di)) = (ax)。

(4)“[bx+si]”可写为“[bx][si]”,“[bx+di]”可写为“[bx][di]”。

2、[bx+si+idata]和[bx+di+idata]的含义

(1)[bx+si+idata]表示一个内存单元,它的偏移地址为(bx)+(si)+idata,也即BX寄存器中的数值加上SI寄存器中的数值加上idata[bx+di+idata]表示一个内存单元,它的偏移地址为(bx)+(di)+idata,也即BX寄存器中的数值加上DI寄存器中的数值加上idata

(2)举例:“mov ax, [bx+si+200]”的含义。

①将一个内存单元的内容送入AX。

②[bx+si+200]这个内存单元的长度为2字节(字单元),存放一个字,内存单元的段地址在DS中,偏移地址为BX中的数值加上SI中的数值加上200。

③数学化的描述为:(ax) = ((ds) * 16 + (bx) + (si) + 200)。

(3)举例:“mov [bx+di+200], ax”的含义。

①将一个内存单元的内容送入[bx+di+200]。

②[bx+di+200]这个内存单元的长度为2字节(字单元),存放一个字,内存单元的段地址在DS中,偏移地址为BX中的数值加上DI中的数值加上200。

③数学化的描述为:((ds) * 16 + (bx) + (di) + 200) = (ax)。

(4)“[bx+si+idata]”的其它几种写法:

①[idata+bx+si],如“mov ax, [idata+bx+si]”。

②idata[bx][si],如“mov ax, idata[bx][si]”。

③[bx].idata[si],如“mov ax, [bx].idata[si]”。

④[bx+idata+si],如“mov ax, [bx+idata+si]”。

⑤[bx][si].idata,如“mov ax, [bx][si].idata”。

(5)“[bx+di+idata]”的其它几种写法:

①[idata+bx+di],如“mov [idata+bx+di], ax”。

②idata[bx][di],如“mov idata[bx][di], ax”。

③[bx].idata[di],如“mov [bx].idata[di], ax”。

④[bx+idata+di],如“mov [bx+idata+di], ax”。

⑤[bx][di].idata,如“mov [bx][di].idata, ax”。

四、不同的寻址方式的灵活应用

1、对内存的寻址方式汇总

2、灵活应用不同的寻址方式案例

(1)编程将datasg段中每个单词的头一个字母改为大写字母,每个单词字符串的长度均为16,如下图所示。

assume cs:codesg, ds:datasg

datasg segment

        db ‘1. file         ’

        db ‘2. edit         ’

        db ‘3. search       ’

        db ‘4. view         ’

        db ‘5. options      ’

        db ‘6. help         ’

datasg ends

codesg segment

start:  mov ax, datasg

        mov ds, ax  ;初始化段寄存器DS,它要存放数据段的段地址

        mov bx, 0  ;bx存放数据段偏移地址

        mov cx, 6   ;一共改6个字母,循环6次即可

    s:  mov al, [bx+3]  ;将源字符串中的首个字母送入al,每轮循环送入1个

        and al, 11011111b ;将al中的字母变成大写形式

        mov [bx+3], al  ;将转为大写的字母送回原位

        loop s

        mov ax, 4c00h

        int 21h

codesg ends

end start

(2)编程将datasg段中每个单词改为大写字母,每个单词字符串的长度均为16,如下图所示。(本例涉及嵌套循环,因此会有不同循环共用CX的问题,解决多重循环问题需使用栈结构)

assume cs:codesg, ds:datasg, ss:stacksg

stacksg segment

        dw 0, 0, 0, 0, 0, 0, 0, 0

stacksg ends

datasg segment

        db ‘ibm             ’

        db ‘dec             ’

        db ‘dos             ’

        db ‘vax             ’

datasg ends

codesg segment

start:  mov ax, datasg

        mov ds, ax  ;初始化段寄存器DS,它要存放数据段的段地址

        mov ax, stacksg

        mov ss, ax  ;初始化段寄存器SS,它要存放栈段的段地址

        mov sp, 16  ;初始化栈顶指针

        mov bx, 0  ;bx存放数据段偏移地址

        mov cx, 4   ;一共改4个单词,每个外层循环改一个单词

  s0:  push cx   ;将外层循环的CX值压入栈中

        mov si, 0   ;si存放目标字母相对于单词首字母的“偏移地址”

        mov cx, 3   ;每个单词长度为3,一次内层循环改一个字母

   s:  mov al, [bx+si] ;将源字符串中的一个字母送入al

        and al, 11011111b ;将al中的字母变成大写形式

        mov [bx+si], al ;将转为大写的字母送回原位

        inc si   ;(si) = (si) + 1

        loop s

        add bx, 16  ;改完一个单词后,bx中的偏移地址指向下一个单词开头

        pop cx   ;将外层循环的CX值从栈中弹出

        loop s0

        mov ax, 4c00h

        int 21h

codesg ends

end start

五、用于内存寻址的寄存器用法及数据的表达

1、用于内存寻址的寄存器用法

(1)只有BX、BP、SI、DI可以用在[...]对内存单元寻址,BX以外的通用寄存器、段寄存器不可以用在[...]中

(2)bx可以与si、di相加,bp可以与si、di相加,除此之外,这几个寄存器之间的运算都是不被允许的。

(3)BX寄存器用于内存寻址时,如不使用段前缀指定,默认段地址为数据段地址;BP寄存器用于内存寻址时,如不使用段前缀指定,默认段地址为栈段地址

2、数据及其位置的表达

(1)对于直接包含在机器指令中的数据,称其为立即数(idata),数据包含在指令中,如“mov ax, 1”。

(2)若指令要处理的数据在寄存器中,在汇编指令中给出相应的寄存器名即可,如“mov ax, bx”。

(3)若指令要处理的数据在内存中,由SA:EA(段地址:偏移地址)确定内存单元

3、指令处理数据的长度

(1)在没有任何指定的情况下,指令处理数据的长度取决于能确定长度的操作数的长度(基本都是寄存器),如下所示。

(2)在没有寄存器参与的内存单元访问指令中,建议用word ptr(字长度)或byte ptr(字节长度)显性地指明所要访问的内存单元的长度,否则,CPU无法得知所要访问的单元是字单元还是字节单元。

六、DIV指令

1、DIV指令的功能和用法

(1)DIV是除法指令,使用DIV做除法的时候,被除数默认放在AX或DX、AX中,除数放在其它寄存器或内存单元中,结果分为商和余数,具体如何放置见下表

被除数的长度为16位

被除数的长度为32位

被除数存放的位置

AX

DX(被除数高16位)和AX(被除数低16位)

除数存放的位置

8位长度的内存单元或寄存器

16位长度的内存单元或寄存器

商存放的位置

AL

AX

余数存放的位置

AH

DX

(2)DIV指令格式:DIV <寄存器或内存单元地址>。

(3)使用DIV指令时,切记提前在默认的寄存器中设置好被除数,且默认寄存器不作别的用处。

2、DIV指令使用举例

(1)一些示例指令:

(2)举例:用DIV指令计算data段中第一个数据除以第二个数据后的结果,商存放在第三个数据的存储单元中。

assume cs:code, ds:data

data segment

        dd 100001

        dw 100

        dw 0

data ends

code segment

start:  mov ax, data

        mov ds, ax   ;初始化段寄存器DS,它要存放数据段的段地址

        mov ax, ds:[0]   ;获取data段中第一个数据

        mov dx, ds:[2]   ;获取data段中第二个数据

        div word ptr ds:[4]  ;DIV指令将除法的商存在AX中

        mov ds:[6], ax   ;商存放在数据段第三个数据的存储单元中

        mov ax, 4c00h

        int 21h

code ends

end start

七、DUP指令

1、DUP指令的功能和用法

(1)DUP指令和db、dw、dd等数据定义伪指令配合使用,用来进行数据的重复定义。

(2)DUP使用格式如下,其中重复数据列表中的数据用逗号分隔,DUP会定义<重复次数>次重复数据列表中的数据

        <数据定义伪指令> <重复次数> DUP (<重复数据列表>)

2、DUP指令使用举例

(1)一些示例指令:

 

(2)举例:采用DUP指令定义一个容量为200个字节的栈段。

stack segment

        db 200 dup (0)

stack ends

课程介绍 第1章 预备知识  1.1 汇编语言的由来及其特点   1 机器语言   2 汇编语言   3 汇编程序   4 汇编语言的主要特点   5 汇编语言的使用领域  1.2 数据的表示和类型   1 数值数据的表示   2 非数值数据的表示   3 基本的数据类型  1.3 习题 第2章 CPU资源和存储器  2.1 寄存器组   1 寄存器组   2 通用寄存器的作用   3 专用寄存器的作用  2.2 存储器的管理模式   1 16位微机的内存管理模式   2 32位微机的内存管理模式  2.3 习题 第3章 操作数的寻址方式  3.1 立即寻址方式  3.2 寄存器寻址方式  3.3 直接寻址方式  3.4 寄存器间接寻址方式  3.5 寄存器相对寻址方式  3.6 基址加变址寻址方式  3.7 相对基址加变址寻址方式  3.8 32位地址的寻址方式  3.9 操作数寻址方式的小结  3.10 习题 第4章 标识符和表达式  4.1 标识符  4.2 简单内存变量的定义   1 内存变量定义的一般形式   2 字节变量   3 字变量   4 双字变量   5 六字节变量   6 八字节变量   7 十字节变量  4.3 调整偏移量伪指令   1 偶对齐伪指令   2 对齐伪指令   3 调整偏移量伪指令   4 偏移量计数器的值  4.4 复合内存变量的定义   1 重复说明符   2 结构类型的定义   3 联合类型的定义   4 记录类型的定义   5 数据类型的自定义  4.5 标号  4.6 内存变量和标号的属性   1 段属性操作符   2 偏移量属性操作符   3 类型属性操作符   4 长度属性操作符   5 容量属性操作符   6 强制属性操作符   7 存储单元别名操作符  4.7 表达式   1 进制伪指令   2 数值表达式   3 地址表达式  4.8 符号定义语句   1 等价语句   2 等号语句   3 符号名定义语句  4.9 习题 第5章 微机CPU的指令系统  5.1 汇编语言指令格式   1 指令格式   2 了解指令的几个方面  5.2 指令系统   1 数据传送指令   2 标志位操作指令   3 算术运算指令   4 逻辑运算指令   5 移位操作指令   6 位操作指令   7 比较运算指令   8 循环指令   9 转移指令   10 条件设置字节指令   11 字符串操作指令   12 ASCII-BCD码调整指令   13 处理器指令  5.3 习题 第6章 程序的基本结构  6.1 程序的基本组成   1 段的定义   2 段寄存器的说明语句   3 堆栈段的说明   4 源程序的结构  6.2 程序的基本结构   1 顺序结构   2 分支结构   3 循环结构  6.3 段的基本属性   1 对齐类型   2 组合类型   3 类别   4 段组  6.4 简化的段定义   1 存储模型说明伪指令   2 简化段定义伪指令   3 简化段段名的引用  6.5 源程序的辅助说明伪指令   1 模块名定义伪指令   2 页面定义伪指令   3 标题定义伪指令   4 子标题定义伪指令  6.6 习题 第7章 子程序和库  7.1 子程序的定义  7.2 子程序的调用和返回指令   1 调用指令   2 返回指令  7.3 子程序的参数传递   1 寄存器传递参数   2 存储单元传递参数   3 堆栈传递参数  7.4 寄存器的保护与恢复  7.5 子程序的完全定义   1 子程序完全定义格式   2 子程序的位距   3 子程序的语言类型   4 子程序的可见性   5 子程序的起始和结束操作   6 寄存器的保护和恢复   7 子程序的参数传递   8 子程序的原型说明   9 子程序的调用伪指令   10 局部变量的定义  7.6 子程序库   1 建立库文件命令   2 建立库文件举例   3 库文件的应用   4 库文件的好处  7.7 习题 第8章 输入输出和中断  8.1 输入输出的基本概念   1 I/O端口地址   2 I/O指令  8.2 中断   1 中断的基本概念   2 中断指令   3 中断返回指令   4 中断和子程序  8.3 中断的分类   1 键盘输入的中断功能   2 屏幕显示的中断功能   3 打印输出的中断功能   4 串行通信口的中断功能   5 鼠标的中断功能   6 目录和文件的中断功能   7 内存管理的中断功能   8 读取和设置中断向量  8.4 习题 第9章 宏  9.1 宏的定义和引用   1 宏的定义   2 宏的引用   3 宏的参数传递方式   4 宏的嵌套定义   5 宏与子程序的区别  9.2 宏参数的特殊运算符   1 连接运算符   2 字符串整体传递运算符   3 字符转义运算符   4 计算表达式运算符  9.3 与宏有关的伪指令   1 局部标号伪指令   2 取消宏定义伪指令   3 中止宏扩展伪指令  9.4 重复汇编伪指令   1 伪指令REPT   2 伪指令IRP   3 伪指令IRPC  9.5 条件汇编伪指令   1 条件汇编伪指令的功能   2 条件汇编伪指令的举例  9.6 宏的扩充   1 宏定义形式   2 重复伪指令REPEAT   3 循环伪指令WHILE   4 循环伪指令FOR   5 循环伪指令FORC   6 转移伪指令GOTO   7 宏扩充的举例   8 系统定义的宏  9.7 习题 第10章 应用程序的设计  10.1 字符串的处理程序  10.2 数据的分类统计程序  10.3 数据转换程序  10.4 文件操作程序  10.5 动态数据的编程  10.6 COM文件的编程  10.7 驻留程序  10.8 程序段前缀及其应用   1 程序段前缀的字段含义   2 程序段前缀的应用  10.9 习题 第11章 数值运算协处理器  11.1 协处理器的数据格式   1 有符号整数   2 BCD码数据   3 浮点数  11.2 协处理器的结构  11.3 协处理器的指令系统   1 操作符的命名规则   2 数据传送指令   3 数学运算指令   4 比较运算指令   5 超越函数运算指令   6 常数操作指令   7 协处理器控制指令  11.4 协处理器的编程举例  11.5 习题 第12章 汇编语言和C语言  12.1 汇编语言的嵌入  12.2 C语言程序的汇编输出  12.3 一个具体的例子  12.4 习题 附录
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Zevalin爱灰灰

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值