++i和i++的区别
- ++i表示取i的地址,增加它的内容,然后把值装入寄存器中
- i++表示取i的地址,把它的值装入寄存器中,然后增加内存中i的值
C代码
int main()
{
int ret = 0;
int i = 2;
ret = i++;
ret = ++i;
return ret;
}
汇编代码
gcc -S 编译上述c代码生成汇编代码如下
.file "assam.c"
.text
.globl main
.type main, @function
main:
.LFB0:
.cfi_startproc
pushq %rbp
.cfi_def_cfa_offset 16
.cfi_offset 6, -16
movq %rsp, %rbp
.cfi_def_cfa_register 6
movl $0, -4(%rbp)
movl $2, -8(%rbp)
movl -8(%rbp), %eax
leal 1(%rax), %edx # i++
movl %edx, -8(%rbp) # i=i+1
movl %eax, -4(%rbp) # ret=i
addl $1, -8(%rbp) # ++i即i=i+1
movl -8(%rbp), %eax
movl %eax, -4(%rbp) # ret=i+1
movl -4(%rbp), %eax
popq %rbp
.cfi_def_cfa 7, 8
ret
.cfi_endproc
.LFE0:
.size main, .-main
.ident "GCC: (GNU) 4.8.5 20150623 (Red Hat 4.8.5-4)"
.section .note.GNU-stack,"",@progbits
解释说明
指令 leal 的作用是load effective address, 本用于地址操作的
leal 1(%rax), %edx 的意思是 rax的值+1 存放到edx
当然这是个技巧了,正规做法得两条指令才能实现上述功能
movl %eax, %edx
addl $1, %edx
当然,此处是gcc进行了优化,其它编译器不一定会这么做