汇编语言之“复制字符串”
汇编语言之“复制字符串”
综合了一下老师上课的内容和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