检测点16.2:将data段中a处的8个数据累加,结果存储到b处的字中,补全程序
mov ax,data
mov es,ax
测试用代码
assume cs:code,es:data
data segment
a db 1,2,3,4,5,6,7,8
b dw 0
data ends
code segment
start: mov ax,0
mov ds,ax
mov ax,data
mov es,ax
mov si,0
mov cx,8
s: mov al,a[si]
mov ah,0
add b,ax
inc si
loop s
mov ax,4c00H
int 21H
code ends
end start
- 在上述源码中,我增加了
mov ax,0 mov ds,ax
,目的是看 a[si] 到底用得是不是es 寄存器
开局就将 ds 置为 0000H
那么如果a[si] 使用的是原先印象里我们认为默认的 ds 寄存器,
a[si]就无法访问到正确的位置,同时累加也得不到正确答案 0024H
另外的,
[1] assume es:data
[2] mov ax,data mov es,ax
这两处代码的结合,按照预期应该是 a[si] 本质等于 es:[offset a+ si]
也就是使用 es 寄存器,从而得到正确累加0024H
换言之,现在的测试用代码如果可以得到 0024H 那么说明,es寄存器起了作用
assume es:data 与 mov ax,data mov es,ax 对于 a[si] 缺一不可
- 1、assume约定 里需要将 es 寄存器 与 data 段练习起来 :
assume es:data
- 2、code段 里要设置 es 寄存器的地址为data段:
mov ax data mov es ax
编译器将标号所表示的地址当做数据的值
(一)
data segment
a db 1,2,3,4,5,6,7,8
b dw 0
c dw a,b
data ends
等价于
c dw a,b
c dw offset a, offset b
数据标号c处存储的两个字型数据为
标号a的偏移地址
标号b的偏移地址
(二)
data segment
a db 1,2,3,4,5,6,7,8
b dw 0
c dd a,b
data ends
等价于
c dd a,b
c dw offset a,seg a,offset b,seg b
数据标号c处存储的两个双字型数据为
标号a的偏移地址和段地址
标号b的偏移地址和段地址
offset 操作符取出某一标号的偏移地址
seg 操作符取出某一标号的段地址
使 a[si] 默认使用段地址寄存器 es
assume es data a[si] 段地址是es.png
奇怪的偏移量来自哪里?
- table 等价于 table[0] 等价于 CS:N[0],如何计算 N?
N = table 的偏移地址 - code segment 后第一个机器码所在偏移地址(code段开始第一个机器码所在的偏移地址)
如上图所示,程序的确由start开始,但是table里的N是相对0000H开始计算的
偏移地址0000H是code段的开始