匯編語言之“復制字符串”
綜合了一下老師上課的內容和mooc里的知識點(疫情期間的痛楚——一堂課上兩次)
主要分為兩個思路,一種是寫循環,可細分為間接尋址和直接尋址;另一種是用指令,也是比較簡單方便的做法
一、前言
我認為有必要先記錄一下字符串的定義方式
分為源字符串和目的字符串
源字符串:
srcmsg db 'fuzhou university',0
目的字符串:
dstmsg db sizeof srcmsg dup(?) \n db 0
在定義字符串時,末尾最好要加上結束符‘0’,但是有時因為dup(?)表示默認全是0,可能就會沒有加,最好養成良好習慣。
二、循環法
上圖為間接尋址法
上圖為相對尋址法
相同點就是都需要寫循環,而我們實際上可以直接用指令代替(這里顯示字符串用的函數是dispmsg,與下文的writestring不同,他使用的寄存器是eax)
三、指令法
代碼釋義:
lea 表示取到msg1的首地址,存入esi,msg2的首地址存入edi(存入的時特定的寄存器,是有實際意義的,后面會說明)
然后用ecx存放msg1的長度,也就是存放復制字符的長度
CLD(CLear Direction flag)則是清方向標志位,也就是使DF值為0,在執行串操作時,使地址按遞增的方式變化,這樣便於調整相關段的的當前指針。這條指令與STD(SeT Direction flag)的執行結果相反,即置DF的值為1
rep movsb 是重復移動復制的意思
此時msg2已經完全復制得到了msg1的字符,由於writestring的輸出需要放在edx寄存器中,所以這里也是mov到特定寄存器edx。
可以說僅僅兩行代碼,cld \n rep movsb \n ,即可以替代一個繁瑣的循環結構
以下為參考資料,可以有助於理解cld \n rep movsb \n是如何替代循環的;from:https://zhidao.baidu.com/question/130424698.html
————————————— ——————— 分割線 ———————— ——————————————
先說說MOVSB(MOVe String Byte):即字符串傳送指令,這條指令按字節傳送數據。通過SI和DI這兩個寄存器控制字符串的源地址和目標地址,比如DS:SI這段地址的N個字節復制到ES:DI指向的地址,復制后DS:SI的內容保持不變。
而REP(REPeat)指令就是“重復”的意思,術語叫做“重復前綴指令”,因為既然是傳遞字符串,則不可能一個字(節)一個字(節)地傳送,所以需要有一個寄存器來控制串長度。這個寄存器就是CX,指令每次執行前都會判斷CX的值是否為0(為0結束重復,不為0,CX的值減1),以此來設定重復執行的次數。因此設置好CX的值之后就可以用REP MOVSB了。
CLD(CLear Direction flag)則是清方向標志位,也就是使DF的值為0,在執行串操作時,使地址按遞增的方式變化,這樣便於調整相關段的的當前指針。這條指令與STD(SeT Direction flag)的執行結果相反,即置DF的值為1。
;example:把當前數據段中偏移1000H開始
;的100個字節數據傳送到偏移2000H開始的
;單元中
cld ;指針增值
push DS ;當前數據段,因此壓入棧中保存
pop ES ;使ES=DS
mov SI,1000H ;源串指針初值
mov DI,2000H ;目的串指針初值
mov CX,100 ;循環次數
Next:
lodsb ;取一個字節
stosb ;存一個字節
loop Next ;循環CX次
;以下程序片段與上面的等價:
cld ;地址指針增值
……
mov CX,100 ;循環CX次
Next:
movsb ;每次傳送一字節數據
loop Next
;或者用更簡單的寫法:
cld
……
mov CX,100
rep movsb