[转]比较汇编指令 LEA 和 MOV

来源:http://www.cnblogs.com/polymorphism/archive/2011/12/12/2285334.html

先介绍几条背景知识:

  1. MOV 的右值必须是常量,而不能是表达式,比如

可以写MOV EAX, EBP,但不能写MOV EAX, EBP + 8

这是因为EBP + 8本身也需要一条指令来计算,所以不能跟MOV写在一条指令里。

  1. 注意到在汇编指令的内存地址符[]内可以做算术运算,那是因为内存地址的计算在CPU里是由专门的处理单元AGU来处理的,并不占用算术运算单元ALU的时钟周期。但如果用MOV 接内存地址符号[]的话,会把[]里的地址指向的内存的内容取出来放入寄存器。

比如 mov eax,[ebx+ecx*4h-20h],会把ebx+ecx*4h-20h计算的结果当成一个内存地址,然后去内存把该地址的内容取出送往eax。

如果我们只是想得到算术运算结果怎么办呢?这时候就可以用到LEA指令了。因为LEA后面接内存地址符[]会把地址,而不是地址里的内容送入寄存器。比如,我们想计算ebx+ecx*4h-20h的结果,就可以这样写:

lea eax,[ebx+ecx*4h-20h]。

当然如果不用lea指令也可以达到目的,不过那样写起来就麻烦多了:

imul ecx, 4

add ebx, ecx

sub ebx, 20h

mov eax, ebx

———-

补充一下,AT&T汇编里面,目的地址和源地址是反的,跟上面对应的汇编是
LEAL -20(%ebx, %ecx, 4), %eax
MOVL -20(%ebx, %ecx, 4), %eax

注意,-20(%ebx, %ecx, 4)表示取计算结果的地址指向的内容。而LEA只取计算结果的地址

updatedupdated2022-02-222022-02-22