汇编语言之“复制字符串”
<h1 id="汇编语言之复制字符串">汇编语言之“复制字符串”</h1><blockquote>
<h3 id="综合了一下老师上课的内容和mooc里的知识点疫情期间的痛楚一堂课上两次">综合了一下老师上课的内容和mooc里的知识点(疫情期间的痛楚——一堂课上两次)</h3>
<p>主要分为两个思路,一种是写循环,可细分为间接寻址和直接寻址;另一种是用指令,也是比较简单方便的做法</p>
</blockquote>
<h2 id="一前言">一、前言</h2>
<blockquote>
<p>我认为有必要先记录一下字符串的定义方式</p>
<p>分为源字符串和目的字符串</p>
<blockquote>
<p>源字符串:<code>srcmsg db 'fuzhou university',0</code></p>
<p>目的字符串:<code>dstmsg db sizeof srcmsg dup(?) \ndb 0</code></p>
<p>在定义字符串时,末尾最好要加上结束符‘0’,但是有时因为dup(?)表示默认全是0,可能就会没有加,最好养成良好习惯。</p>
</blockquote>
</blockquote>
<h2 id="二循环法">二、循环法</h2>
<p><img src="https://img2020.cnblogs.com/blog/1833384/202003/1833384-20200309114154538-1242014721.png" alt="" loading="lazy"></p>
<h3 id="上图为间接寻址法">上图为间接寻址法</h3>
<p><img src="https://img2020.cnblogs.com/blog/1833384/202003/1833384-20200309120300664-1599679207.png" alt="" loading="lazy"></p>
<h3 id="上图为相对寻址法">上图为相对寻址法</h3>
<h3 id="相同点就是都需要写循环而我们实际上可以直接用指令代替这里显示字符串用的函数是dispmsg与下文的writestring不同他使用的寄存器是eax">相同点就是都需要写循环,而我们实际上可以直接用指令代替(这里显示字符串用的函数是dispmsg,与下文的writestring不同,他使用的寄存器是eax)</h3>
<h2 id="三指令法">三、指令法</h2>
<p><img src="https://img2020.cnblogs.com/blog/1833384/202003/1833384-20200309120402032-566967229.png" alt="" loading="lazy"></p>
<blockquote>
<p>代码释义:</p>
<p>lea 表示取到msg1的首地址,存入esi,msg2的首地址存入edi(存入的时特定的寄存器,是有实际意义的,后面会说明)</p>
<p>然后用ecx存放msg1的长度,也就是存放复制字符的长度</p>
<p>CLD(CLear Direction flag)则是清方向标志位,也就是使DF值为0,在执行串操作时,使地址按递增的方式变化,这样便于调整相关段的的当前指针。这条指令与STD(SeT Direction flag)的执行结果相反,即置DF的值为1</p>
<p>rep movsb 是重复移动复制的意思</p>
<p>此时msg2已经完全复制得到了msg1的字符,由于writestring的输出需要放在edx寄存器中,所以这里也是mov到特定寄存器edx。</p>
<h3 id="可以说仅仅两行代码cld---n---rep-movsb--n----即可以替代一个繁琐的循环结构">可以说仅仅两行代码,cld \n rep movsb\n ,即可以替代一个繁琐的循环结构</h3>
</blockquote>
<p>以下为参考资料,可以有助于理解cld \n rep movsb\n是如何替代循环的;from:https://zhidao.baidu.com/question/130424698.html</p>
<p>————————————— ——————— 分割线 ———————— ——————————————</p>
<p>先说说MOVSB(MOVe String Byte):即字符串传送指令,这条指令按字节传送数据。通过SI和DI这两个寄存器控制字符串的源地址和目标地址,比如DS:SI这段地址的N个字节复制到ES:DI指向的地址,复制后DS:SI的内容保持不变。</p>
<p>而REP(REPeat)指令就是“重复”的意思,术语叫做“重复前缀指令”,因为既然是传递字符串,则不可能一个字(节)一个字(节)地传送,所以需要有一个寄存器来控制串长度。这个寄存器就是CX,指令每次执行前都会判断CX的值是否为0(为0结束重复,不为0,CX的值减1),以此来设定重复执行的次数。因此设置好CX的值之后就可以用REP MOVSB了。</p>
<p>CLD(CLear Direction flag)则是清方向标志位,也就是使DF的值为0,在执行串操作时,使地址按递增的方式变化,这样便于调整相关段的的当前指针。这条指令与STD(SeT Direction flag)的执行结果相反,即置DF的值为1。</p>
<p>;example:把当前数据段中偏移1000H开始<br>
;的100个字节数据传送到偏移2000H开始的<br>
;单元中</p>
<p>cld ;指针增值<br>
push DS ;当前数据段,因此压入栈中保存<br>
pop ES ;使ES=DS<br>
mov SI,1000H ;源串指针初值<br>
mov DI,2000H ;目的串指针初值<br>
mov CX,100 ;循环次数<br>
Next:<br>
lodsb ;取一个字节<br>
stosb ;存一个字节<br>
loop Next ;循环CX次</p>
<p>;以下程序片段与上面的等价:</p>
<p>cld ;地址指针增值<br>
……<br>
mov CX,100 ;循环CX次<br>
Next:<br>
movsb ;每次传送一字节数据<br>
loop Next</p>
<p>;或者用更简单的写法:<br>
cld<br>
……<br>
mov CX,100<br>
rep movsb</p>
</div>
<div id="MySignature" role="contentinfo">
<div>作者:breeze</div>
<div>出处:https://www.cnblogs.com/fragrant-breeze/</div>
<div>本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须在文章页面给出原文连接,否则保留追究法律责任的权利。 </div><br><br>
来源:https://www.cnblogs.com/fragrant-breeze/p/12447960.html
頁:
[1]