汇编语言(七)--更灵活的定位内存地址的方法

第七章 更灵活的定位内存地址的方法

7.1 and和or指令

and指令

逻辑与指令,按位进行与运算。
如:

mov al, 01100011B
and al, 00111011B
执行后:al = 00100011B

通过该指令可将操作对象的相应位设为0,其他位不变。

7.2 关于ASCII码

ASCII编码,是在计算机系统中通常被采用的。
简单地说,所谓编码方案,就是一套规则,它约定了用什么样的信息来表示现实对象。

7.3 以字符形式给出数据

我们可以在汇编程序中,用 “……”的方式指明数据是以字符的形式给出的,编译器将把它们转化为相对应的ASCII码。

assume ds:data
data segment
 db 'unIX' 
 db 'foRK'
data ends
code segment
  start:mov al,'a'
        mov bl,'b'
        mov ax,4c00h
        int 21h
code ends
end start 

上面的源程序中:

  1. “db ‘unIX’ ” 相当于“db 75H,6EH,49H,58H”, “u”、 “n”、 “I”、 “X”的ASCII码分别为75H、6EH、49H、58H;
  2. “db ‘foRK’ ” 相当于“db 66H,6FH,52H,4BH”, “f”、 “o”、 “R”、 “K”的ASCII码分别为66H、6FH、52H、4BH;
  3. “mov al,’a’”相当于“mov al,61H”,”a”的ASCII码为61H;
  4. “mov al,’b’”相当于“mov al,62H”,”b”的ASCII码为62H。

7.4 大小写转换问题

我们可以知道,小写字母的ASCII码值比大写字母的ASCII码值大20H 。

将”BaSiC“改成全部大写

assume cs:codesg,ds:datasg
datasg segment
db 'BaSiC'
db 'iNfOrMaTiOn'
datasg ends
codesg segment
  start: mov ax,datasg
       mov ds,ax
       mov bx,0
       mov cx,5
    s: mov al,[bx]
       如果(al)>61H,则为小写字母ASCII码,则:sub al,21H
       mov [bx],al
       inc bx
       loop s
:
codesg ends
end start

7.5 [bx+idata]

一种更为灵活的方式来指明内存单元:[bx+idata]表示一个内存单元,它的偏移地址为(bx)+idata(bx中的数值加上idata)

我们看一下指令mov ax,[bx+200]的含义:
将一个内存单元的内容送入ax,这个内存单元的长度为2字节(字单元),存放一个字,偏移地址为bx中的数值加上200,段地址在ds中
数学化的描述为: (ax)=((ds)*16+(bx)+200)

另外常用的表示方法:

mov ax,[200+bx]
mov ax,200[bx]
mov ax,[bx].200

7.6 用[bx+idata]的方式进行数组的处理

简而言之就是,对于两个起点不同需要进行循环的数组来说,他们的变化一定是同步的。所以这种方法下只需要一个循环就可以满足题目的需求

7.7 SI和DI

SI和DI是8086CPU中和bx功能相近的寄存器,SI和DI不能够分成两个8 位寄存器来使用。

下面的三组指令实现了相同的功能:

(1) mov bx,0;mov ax,[bx]
(2) mov si,0;mov ax,[si]
(3) mov di,0;mov ax,[di]

问题: 用寄存器SI和DI实现将字符串‘welcome to masm!’复制到它后面的数据区中

  1. 现在我们要对datasg 段中的数据进行复制,我们先来看一下要复制的数据在什么地方,datasg:0,这是要进行复制的数据的地址。
  2. “welcome to masm!”从偏移地址0开始存放,长度为 16 个字节,所以,它后面的数据区的偏移地址为 16 ,就是字符串所要存放的空间。
  3. “welcome to masm!”从偏移地址0开始存放,长度为 16 个字节,所以,它后面的数据区的偏移地址为 16 ,就是字符串所要存放的空间。
codesg segment
start: mov ax,datasg
         mov ds,ax
         mov si,0
         mov di,16
         mov cx,8
    s:   mov ax,[si]
         mov [di],ax
         add si,2
         add di,2
         loop s

         mov ax,4c00h
         int 21h
codesg ends
end start

我们可以利用[bx(si或di)+idata]的方式,来使程序变得简洁。

codesg segment
start: mov ax,datasg
         mov ds,ax
         mov si,0
         mov cx,8
    s:   mov ax,0[si]
         mov 16[si],ax
         add si,2
         loop s
         mov ax,4c00h
         int 21h
codesg ends
end start

7.8 [bx+si]和[bx+di]

在前面,我们用[bx(si或di)]和[bx(si或di)+idata] 的方式来指明一个内存单元,我们还可以用更灵活的方式:

[bx+si] /[bx+di]

我们看下指令mov ax,[bx+si]的含义:
将一个内存单元的内容送入ax,这个内存单元的长度为2字节(字单元),存放一个字,偏移地址为bx中的数值加上si中的数值,段地址在ds中。

指令mov ax,[bx+si]的数学化的描述为:(ax)=( (ds)*16+(bx)+(si) )
该指令也可以写成如下格式(常用):mov ax,[bx] [si]

7.9 [bx+si+idata]和[bx+di+idata]

[bx+si+idata]表示一个内存单元,它的偏移地址为(bx)+(si)+idata

指令mov ax,[bx+si+idata]的含义:
将一个内存单元的内容送入ax,这个内存单元的长度为2字节(字单元),存放一个字,偏移地址为bx中的数值加上si中的数值再加上idata,段地址在ds中

7.10 不同的寻址方式的灵活应用

如果我们比较一下前而用到的几种定位内存地址的方法(可称为寻址方式),就可以发现有以下几种方式:
(1)[iata] 用一个常量来表示地址,可用于直接定位一个内存单元;
(2)[bx]用一个变量来表示内存地址,可用于间接定位一个内存单元;
(3)[bx+idata] 用一个变量和常量表示地址,可在一个起始地址的基础上用变量间接定位一个内存单元;
(4)[bx+si]用两个变量表示地址;
(5)[bx+si+idata] 用两个变量和一个常量表示地址。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值