匯編語言學習assume的作用詳解

assume 的作用是關聯段名與段寄存器。

如果你在數據段中定義瞭變量名,比如:

x   db  0

而你在代碼中,需要直接使用這個變量名,比如:

mov   al,  x

那麼,匯編程序在匯編時,就會報告錯誤。

因為,mov指令中遇到 x 這個變量名時,匯編程序不知道它要用哪個段寄存器作為段地址。

所以:

若要用變量名直接訪問,或使用語句標號(比如你例子中的標號 start)就必須要在assume偽指令中將這些變量或標號所在段的段名,與段寄存器名關聯,否則會出錯。

如果你不使用段中的變量名,可以不關聯這個段的段名與寄存器。

如果你訪問變量時,都指定瞭段跨越前綴,關聯也不是必須的。比如你可以用 mov al, ds:x訪問變量 x 。

這幾天在看王爽大大的 匯編語言。對於assume偽指令卻很是不懂。

比如已經定義瞭assume cs:code,ds:data

但用debug觀察的時候,發現ds段寄存器卻沒有相關聯的數據。

必須在cs中寫明: mov ax,data

                            mov ds,ax
然後才能發現ds中有正確的數據。

於是疑惑,assume不是已經關聯瞭ds嘛?

上網求助 = =、 然後找到答案。

編寫程序,是寫給編譯軟件的。由編譯軟件,編譯成機器碼,再去控制CPU。但是,編譯軟件,對assume語句,並不生成機器碼。

所以,必須有mov ax,data,mov ds,ax,CPU才能受控。

---assume語句,是偽指令,僅僅是寫給編譯軟件的。編譯軟件,並不把它生成機器碼。

assume對除瞭CS以外的其它段寄存器,僅僅隻是關聯瞭段名,以便在訪問段內變量時程序可以知道用哪個段寄存器,並沒有在程序加載時將段地址裝入段寄存器。

所以,將段地址裝入段寄存器的工作,必須由用戶在程序中自己編寫代碼,並在程序開始運行時執行代碼完成裝入工作。
僅僅對CS段寄存器,會在關聯段名的同時,在程序加載時自動將段地址裝入段寄存器。
—-補充:前天知道瞭答案後,我以為assume ds:data 之類的指沒有什麼用,隻是給程序員看的。

但今天發現不是這樣的。 如果你在data中用瞭標號的話,則assume ds:data不能省略。

比如:

data segment
a db 1,2,3,4 ,5,6,7,8
b dw 0
data ends

a,b的後面沒有“ :”。

如果你想在cs段中用數據標號訪問數據,則必須在開頭加上assume ds:data,否則會報錯

Arror A2068:Can not address with segment register

不過就算在開頭加上瞭assume ds:data,代碼段中也不能少瞭mov ax,data,mov ds,ax。

作用:用於標識默認段前綴

解釋:assume 並不能改變ds等段寄存器的值,但他能改變編譯器產生的匯編代碼。比如:

assume ss:stack

stack segment

x :db 0

stack ends

如果程序需要mov ax,[x],那麼程序如何定位[x]呢?我們知道x隻是一個偏移地址0,所以此時assume就相當於告訴編譯器stack段的所有標號都與ss相關聯,所以此時[x]就相當於ss:[0].如果我們直接將這句改為mov ax,ss:[0],那麼前面不加assume也是可以的.這也是為什麼[0]被編譯器強制理解為立即數,而[標號]卻被理解為標號裡的內容的原因,因為標號必須與段assume,否則會報錯cannot address with segment register.而[0]無默認段,就隻能被認為為立即數瞭.

所以,我們仍需在程序中將ss的值,用指令修改為stack,原因就是assume並不會修改段寄存器,這個由dos系統決定,如果dos系統決定將段值編譯進.exe文件頭,並在加載進內存時根據文件頭,修改段值,那麼此時assume就相當於可以改變段值瞭.但是我調試的現實是ds、es指向psp頭(psp詳見16位exe程序加載過程),ss指向ds+0:00f0,cs指向ds+0:0100。

以上就是匯編語言學習assume的作用詳解的詳細內容,更多關於匯編語言assume作用的資料請關註WalkonNet其它相關文章!

推薦閱讀: