来源:http://www.cnblogs.com/polymorphism/archive/2011/12/12/2285334.html
先介绍几条背景知识:
- MOV 的右值必须是常量,而不能是表达式,比如
可以写MOV EAX, EBP,但不能写MOV EAX, EBP + 8
这是因为EBP + 8本身也需要一条指令来计算,所以不能跟MOV写在一条指令里。
- 注意到在汇编指令的内存地址符[]内可以做算术运算,那是因为内存地址的计算在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只取计算结果的地址