关于stosb字节传输目的地址问题
若在一片内存区域中有一数据块,起始地址为num。数据块中的数有正数有负数,要求把其中的正负数分开,分别放在同一段的两个缓冲区存放正数的缓冲区起始地址为plus_n,存放负数的缓冲区起始地址为minus_n。执行后,plus_n 和minus_n处存储的数都是从偏移地址2处开始存储的,开始两个字节分别为10 00和00 00,这是为何?
data segment
num db 01h,10h,88h,0a4h,01eh,90h,40h,85h
db 0e8h,4fh,3ch,86h,87h,30h,20h,81h
count dw $-offset num
plus_n db 16 dup(?)
minus_n db 16 dup(?)
data ends
assume cs:code,ds:data,es:data
code segment
start: mov ax,data
mov ds,ax
mov es,ax
lea si,offset num
lea di,offset plus_n
mov bx,offset minus_n
mov cx,count
cld
go_on: lodsb
test al,80h
jnz min_s
stosb
jmp again
min_s: xchg di,bx
stosb
xchg di,bx
again: loop go_on
mov ax,4c00h
int 21h
code ends
end start 本帖最后由 jackz007 于 2025-3-28 13:43 编辑
count dw $-offset num
这样定义 count 会实际占用 2 个字节,10 00 就是 count 的数值,所以,plus_n 和 minus_n 并不是从 076A : 0010 和 076A : 0020 处开始的,而是从 076A : 0012 和 076A : 0022 处开始的。现在再看你的数据,是不是就合理了?
要是想要自己预想的结果,那么,count 应该这样定义:
count equ $ - offset num
这样定义的 count 就不会占用内存了。 非常感谢,因为对两种定义方式不懂,只是模仿别人的程序。 执行后,plus_n 和minus_n处存储的数都是从偏移地址2处开始存储的,开始两个字节分别为10 00和00 00,这是为何?
问题分析:
问题根源是数据段布局理解偏差。观察以下关键点:
1. count dw $-offset num 占用2字节,位于num数据块之后
2. plus_n和minus_n的实际起始地址在count之后
3. 用户可能误将count的存储空间当成了缓冲区的一部分
内存布局详解:
假设数据段从偏移地址0000h开始:
[*]num:0000h~000Fh(16字节)
[*]count:0010h~0011h(值为0010h,小端存储为10 00)
[*]plus_n:0012h~0021h(16字节)
[*]minus_n:0022h~0031h(16字节)
现象解释:
1. plus_n的真实起始地址是0012h,但若误认为起始地址是0010h:
- 0010h~0011h实际是count的值10 00
- 真实数据从0012h开始,却被误认为是"偏移地址2处"
2. minus_n同理:
- 若误认为起始地址是0020h
- 实际起始地址是0022h
- 0020h~0021h可能残留00(未初始化内存)
代码改进建议:
修改指针初始化方式更规范:
start:mov ax,data
mov ds,ax
mov es,ax
mov si,offset num ; 明确用mov加载偏移地址
mov di,offset plus_n
mov bx,offset minus_n
验证方法:
使用调试工具(如Debug)查看内存:
[*]数据段地址 = DS:
[*]num数据 = DS:
[*]count值 = DS:
[*]真实plus_n数据 = DS:
[*]真实minus_n数据 = DS:
正确执行后,正数缓冲区应包含:01h,10h,1Eh,40h,4Fh,3Ch,30h,20h;负数缓冲区应包含:88h,A4h,90h,85h,E8h,86h,87h,81h。
以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
页:
[1]