标题: 清华的《IBM-PC汇编程序设计》误导我?
[打印本页]
作者: Loff
时间: 2004-6-4 00:00
标题: 清华的《IBM-PC汇编程序设计》误导我?
相信许多人学习dos汇编都是看这本书吧?
里面介绍寻址方式时,解释mov ax,[2000H]一句为:将ds:[2000h]处的内容赋给ax,即,若ds:[2000H]处的值为3020,则执行这一句后,ax =3020H。
近日调试dos程序,发现形如mov xx,[XXXXh]、add xx,[XXXXh](里面都是数字)的指令都被汇编程序解释为mov xx,xxxxh,即直接将中括号里面的值给了寄存器!也就是说,mov ax,[2000H]执行后, ax=2000H。不信你可以自己编几句,然后装入调试器来查看内存里的代码,会发现执行改句后ax=2000h。
除非是我的机器有问题,或者编程软件有问题(masm6.11、tasm5、Tdebug)。不然请大家告诉我应该如何理解这种寻址方式。
作者: 拉菲尔
时间: 2004-6-10 00:00
尽信书,不如无书
作者: 晓欣
时间: 2004-6-11 00:00
没有吧,是你自己没有写对吧
作者: Loff
时间: 2004-6-12 00:00
.386c
cseg segment byte public use16
assume cs:cseg, ds:dseg, ss:sseg
start:
mov ax,dseg
mov ds,ax
lop:
xor si,si ;清零si
mov word ptr ds:[0C8h],0 ;内存ds:[0C8h]处的值清零
cmp si,[0C8h] ;si的值等于ds:[0C8h]处的值么(0值)
je _exit ;如果等则退出程序
jmp lop ;否则循环判断。
_exit:
mov ax, 4C00h
int 21h
cseg ends
;---------------------------------------------------
dseg segment byte public use16
db 0 dup(1024)
dseg ends
;----------------------------------------------
sseg segment stack
db 100h dup(?)
sseg ends
;----------------------------------------------
end start
程序执行的结果是死循环,而按照[xxxx]是内存地址内容的理解,应该是正常退出程序。用tdb跟踪调试发现cmp一句被编译为 cmp si,00C8,也就是说,[0C8h]被编译器认为是立即数0C8h,而不是代表[0C8h]处存放的值。
再一个例子:
mov [XXXX],ax ; XXXX是一个数值地址,比如1234h。
编译报错:error A2001: immediate operand not allowed(不允许立即方式操作数)
也就是说,编译器将[XXXX]认为是立即数,而不是一个内存地址。
我改为mov ds:[XXXX],ax 就编译通过了。而且正确地存入了该地址。
编译器版本MASM6.11
[此贴子已经被作者于2004-6-12 上午 09:57:35编辑过]
作者: boblhh9999
时间: 2004-7-1 00:00
不会吧,那岂不是同立即数一样了
作者: Arbiter
时间: 2004-7-14 00:00
小伙子,多找找自己程序的问题吧!
作者: Loff
时间: 2004-7-14 00:00
搞清楚了,在源代码里面,形如[1234]的形式,都被masm编译器识别为立即数(和书上讲的不同)。只有[寄存器]、[符号地址]才是对应的内存值。但是,在调试器里面看到的[1234]却是内存值,与源代码又不同。
作者: buffon
时间: 2006-5-30 10:58
书没问题,加上段前缀就ok了。
作者: pijiudu
时间: 2009-3-11 22:02
书上说的没错,可以看看王爽的书。通俗易懂。
作者: cabinsummer
时间: 2009-3-11 22:09
我刚试了,书上的没错。楼主多找找自己的原因
作者: Qtwdftxvt
时间: 2009-3-12 11:48
看书得取其意,不是仅仅观其形。
作者: netwinxp
时间: 2009-3-12 21:01
1、你的数据段里面声明的是BYTE,用的却是WORD,所以需要加上WORD PTR加以限制。
2、mov 内存,立即数 是可以的,但它不是8086指令集,所以要在程序最前面加.386伪指令,并且要声明内存变量的类型,比如mov word ptr [0C8h],0。
作者: MarsKeeper
时间: 2009-4-28 03:13
mov ax,[2000H]在TASM上的结果与MASM相同。AX的内容为2000H。
12楼的两条理由都是错的,举的例子是对的。