【新书推荐】7.6节 相对基址加变址寻址方式

2024-02-12 16:20

本文主要是介绍【新书推荐】7.6节 相对基址加变址寻址方式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本节内容:基址加变址寻址方式。

         ■基址加变址寻址方式:指令操作数为内操作数,操作数地址使用[基址寄存器+变址寄存器]表示。

7.6.1 基址加变址寻址方式

基址加变址寻址方式的操作数在存储器中,操作数的有效地址由基地址寄存器(BX,BP)之一与变址寄存器(SI,DI)之一相加得到,如图7-12所示。

通常BX、SI、DI引用DS,BP引用SS。

图7-12 基址加变址地址表达式

如果所得地址超出FFFFH,则取其64K(FFFFH)的模。

如图7-13所示:

                                                        图7-13 基址加变址寻址方式

假设DS=5000H,BX=3456H,DI=1234H

MOV AX,[BX+DI]语句中的[BX+DI]内存操作数的物理地址为5468AH,AX=5678H。

动手实验22:演示基址加变址寻址方式

如图7-14所示,在debug调试器内输入:

mov ax,5000H

mov ds,ax;将段值5000H赋给ds段寄存器。

Mov bx,3456H;

Mov di,1234H;

Mov ax,[bx+di];将数据段偏移地址[bx+di]处存储的值送入ax寄存器。

图7-14 演示基址加变址寻址方式

在其他段中使用基址加变址寻址方式:

MOV DS:[BP+SI],AL   ;将AL寄存器的值存入数据段[BP+SI]偏移地址处。

MOV AX,ES:[BX+SI]   ;将附加段ES内偏移地址[BX+SI]处的数值送入AX寄存器。

基址加变址寻址方式通常用于数组或表格处理。用基址寄存器存放数组首地址,而变址寄存器用来定位数组中的元素,或相反。由于两个寄存器都可以改变,所以能更加灵活地访问数组或表格中的元素。

提示

如下两种方式等价:

    MOV AX , [BX+DI]

    MOV AX,[BX][DI]    

7.7节 相对基址加变址寻址方式

         本节内容:相对基址加变址寻址方式。

         ■相对基址加变址寻址方式:指令操作数为内存操作数,操作数的地址采用[基址寄存器+变址寄存器+位移量]的方式表示。示例代码address.asm。

7.7.1 相对基址加变址寻址方式

相对基址加变址寻址方式的操作数在存储器中,操作数的有效地址由基地址寄存器(BX,BP)之一与变址寄存器(SI,DI)之一的内容及指令中给定的8位或16位位移量相加得到,如图7-15所示。

通常BX,SI,DI引用DS,BP引用SS。

图7-15 相对基址加变址地址表达式

在指令中给定的8位或16位位移量如果是有符号整数,则采用补码形式表示。如果是8位有符号整数,则带符号位扩展成16位。

如果所得地址超出FFFFH,则取其64K(FFFFH)的模。

如图7-16所示:

图7-16 相对基址加变址寻址方式

例:假设(DS)=5000H,(BX)=3456H,[DI]=1234H。

那么MOV AX,[BX+DI-2]  ;物理地址为54688H,(AX)=6789H。

动手实验23:演示相对基址加变址寻址方式

如图7-17所示,在debug调试器内输入:

mov ax,5000H

mov ds,ax;将段值5000H赋给ds段寄存器。

Mov word ptr [4688H],6789H;将6789H存入数据段偏移地址4688H处。

Mov bx,3456H;

Mov di,1234H;

Mov ax,[bx+di-2];将数据段偏移地址[bx+di-2]处存储的值送入ax寄存器。

图7-17 相对基址加变址寻址方式

举例

         如表7-1所示,我们在数据定义中定义了一张学生统计表table,表内记录了学生student0开始的若干个学生统计信息。分别统计该名学生的姓名(8个字节)、年龄(1个字节)、学号(1个字节)、身高(2个字节)和体重(2个字节)。如果把这张表存入计算机的线性内存,每名学生需要14个字节的内存空间,且连续存储,如图7-18所示。

接下来我们使用合适的寻址方式来获取table表中的学习信息。

         Table表的起始地址:mov bx,offset table;使用bx基址寄存器。

         学生student0的地址:mov si,offset student0。

学生student1的地址:mov di,offset student1。

学生student0的姓名位于表table内的偏移地址[SI]处,或者表示为偏移地址[bx]。

         学生student0的年龄位于表table内的偏移地址8处,或者表示为偏移地址[bx+8]。

学生student1的姓名位于表table内的偏移地址[di]处,表示为偏移地址[bx+di]。

         学生student1的年龄位于表table内的偏移地址[di +8]处,表示为偏移地址[bx+di+8]。

         如果继续使用变址寄存器SI,则表示如下:

         add si,14;将SI的值加上14个字节,指向学生student1。

学生student1的姓名位于表table内的偏移地址[si]处,表示为偏移地址[bx+ si]。

         学生student1的年龄位于表table内的偏移地址[si +8]处,表示为偏移地址[bx+ si +8]。

表7-1 相对基址加变址寻址方式

图7-18 内存中的学生统计表

尽管这种相对基址加变址寻址方式最复杂,但是也最灵活,其他6种寻址方式都是其变形。

提示

如下四种方式等价:

    MOV AX , [BX+DI+1234H]

    MOV AX , 1234H[BX+DI]

    MOV AX,1234H[BX][DI] 

    MOV AX,1234H[DI][BX]    

动手实验24:寻址方式

编写下面的示例代码address.asm源程序,完成编译链接,生成address.exe可执行程序,并在debug调试器中单步跟踪执行。仔细观察不同寻址方式计算的偏移地址。

示例代码2

;程序名:address

;功能:寻址方式

;============================

assume cs:code,ds:data                             

                                    

X = 1234H                  ;X EQU 1234H          

data segment                             

val1 db 1

val2 dw ?

val3 db 1,2,3,4,5,6,7,8,9,0,1,2,3

val4 dd 12345678h                             

data ends                            

code segment                            

start:                                   

         mov ax,data     ;将段标号data送入ax          

         mov ds,ax         ;ds=ax               

         ;立即寻址方式                          

         mov ax,X                             ;ax=1234H

         mov word ptr val1,3456h;将3456H写入val1偏移地址处,val1=56h,val2=0034h

         ;寄存器寻址方式

         mov cx,ax         ;cx=ax

         ;直接寻址方式

         mov al,byte ptr ds:[val1] ;将val1偏移地址处一个字节的值送入al,al=56h

         mov al,val1                          ;等同于上一条语句的简化形式

         ;寄存器间接寻址方式

         mov si,offset val3              ;将val3偏移地址送入si,si=3

         lea si,word ptr ds:[val3]   ;将val3偏移地址送入si,si=3

         add si,1    ;si=4

         mov bx,word ptr ds:[si]    ;将si偏移地址处一个字的值送入bx=0302h      

         ;寄存器相对寻址方式

         mov bx,offset val1   ;将val1偏移地址送入bx,bx=0

         mov ax,ds:[bx+3]      ;将数据段bx+3偏移地址处一个字的值送入ax,ax=0201h

         ;基址加变址寻址方式

         mov si,1                      ;si=1

         mov bx,offset val3   ;将val3偏移地址送入bx,bx=3

         mov ax,[bx+si]           ;将[bx+si]=4偏移地址处的一个字的值送入ax,ax=0302h

         ;相对基址加变址寻址方式

         mov ax,[bx+si+1]      ;将[bx+si+1]=5偏移地址处的一个字的值送入ax,ax=0403h

         ;

         lea si,val4                            ;将val4偏移地址送入si,si=10H

         mov si,offset val4             ;将val4偏移地址送入si,si=10H

         mov al,byte ptr val4        ;将val4偏移地址处一个字节的值送入al,al=78h

         mov al,byte ptr ds:[si]      ;将si偏移地址处一个字节的值送入al,al=78h

         ;将val4偏移地址处5678H送入di,1234H送入ds

         lds di,val4                             ;ds=1234h,di=5678h

         ;切换回原数据段

         mov ax,data

         mov ds,ax

         ;

         mov al,0                     ;4C号功能入口参数

         mov ah,4ch                ;调用DOS 4C号功能,退出程序

         mov ax,4c00h           ;与前两条语句等价

         int 21h                                 

code ends                                    

end start                            

         编译后加载debug调试器后的反汇编代码如图7-19所示:

图7-19 address.asm反汇编代码

         示例代码2给出了7种不同的寻址方式的示例。Debug调试器单步跟踪时,请读者仔细核对注释中给出的偏移地址是否正确。只有给出正确的地址,才可以正确的读写内存。我们可以简单将计算机的工作原理理解为查数、查表、查地址。随着学习的深入,请读者慢慢理解。

同时需要注意lea指令和mov指令的区别,lea指令取地址,而mov指令取值。Lds指令为复合指令,将四个字节的源操作数中的高字送入ds,低字送入di。我们随后将在下一章详细讲解8086计算机指令。  

练习

1、8086的基本寻址方式分为哪三类?

2、8086存储器寻址方式可分为哪几种?

3、请说明如下指令中源操作数的寻址方式:

MOV BX,[1234H]

MOV BX,1234

MOV DX,BX

MOV DX,[BX]

MOV DX,[BX+1234H]

MOV DX,[BX+DI]

MOV DX,[BX+DI+1234H]

4、如何选择恰当的选择寻址方式?

5、为什么目标操作数不能采用立即寻址方式?

6、哪些存储器寻址方式可能导致有效地址超出64KB的范围?8086是如何处理的?

本文摘自编程达人系列教材《X86汇编语言基础教程》。

这篇关于【新书推荐】7.6节 相对基址加变址寻址方式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/703009

相关文章

Linux线程之线程的创建、属性、回收、退出、取消方式

《Linux线程之线程的创建、属性、回收、退出、取消方式》文章总结了线程管理核心知识:线程号唯一、创建方式、属性设置(如分离状态与栈大小)、回收机制(join/detach)、退出方法(返回/pthr... 目录1. 线程号2. 线程的创建3. 线程属性4. 线程的回收5. 线程的退出6. 线程的取消7.

golang程序打包成脚本部署到Linux系统方式

《golang程序打包成脚本部署到Linux系统方式》Golang程序通过本地编译(设置GOOS为linux生成无后缀二进制文件),上传至Linux服务器后赋权执行,使用nohup命令实现后台运行,完... 目录本地编译golang程序上传Golang二进制文件到linux服务器总结本地编译Golang程序

Linux下删除乱码文件和目录的实现方式

《Linux下删除乱码文件和目录的实现方式》:本文主要介绍Linux下删除乱码文件和目录的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux下删除乱码文件和目录方法1方法2总结Linux下删除乱码文件和目录方法1使用ls -i命令找到文件或目录

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

Linux在线解压jar包的实现方式

《Linux在线解压jar包的实现方式》:本文主要介绍Linux在线解压jar包的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux在线解压jar包解压 jar包的步骤总结Linux在线解压jar包在 Centos 中解压 jar 包可以使用 u

Jenkins分布式集群配置方式

《Jenkins分布式集群配置方式》:本文主要介绍Jenkins分布式集群配置方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1.安装jenkins2.配置集群总结Jenkins是一个开源项目,它提供了一个容易使用的持续集成系统,并且提供了大量的plugin满

Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式

《Java通过驱动包(jar包)连接MySQL数据库的步骤总结及验证方式》本文详细介绍如何使用Java通过JDBC连接MySQL数据库,包括下载驱动、配置Eclipse环境、检测数据库连接等关键步骤,... 目录一、下载驱动包二、放jar包三、检测数据库连接JavaJava 如何使用 JDBC 连接 mys

C#读写文本文件的多种方式详解

《C#读写文本文件的多种方式详解》这篇文章主要为大家详细介绍了C#中各种常用的文件读写方式,包括文本文件,二进制文件、CSV文件、JSON文件等,有需要的小伙伴可以参考一下... 目录一、文本文件读写1. 使用 File 类的静态方法2. 使用 StreamReader 和 StreamWriter二、二进

Knife4j+Axios+Redis前后端分离架构下的 API 管理与会话方案(最新推荐)

《Knife4j+Axios+Redis前后端分离架构下的API管理与会话方案(最新推荐)》本文主要介绍了Swagger与Knife4j的配置要点、前后端对接方法以及分布式Session实现原理,... 目录一、Swagger 与 Knife4j 的深度理解及配置要点Knife4j 配置关键要点1.Spri

Qt QCustomPlot库简介(最新推荐)

《QtQCustomPlot库简介(最新推荐)》QCustomPlot是一款基于Qt的高性能C++绘图库,专为二维数据可视化设计,它具有轻量级、实时处理百万级数据和多图层支持等特点,适用于科学计算、... 目录核心特性概览核心组件解析1.绘图核心 (QCustomPlot类)2.数据容器 (QCPDataC