- 论坛徽章:
- 0
|
这里实现的语义是一样的,只不过不同的编译器做法不同,我理解的GCC可能稍微只能一些。
在乘法的子函数里面,操作的对象实际上是调用它的函数(main)里面的局部变量。 我把dump出来的汇编贴上来
- int main() {
- 8048754: 8d 4c 24 04 lea 0x4(%esp),%ecx
- 8048758: 83 e4 f0 and $0xfffffff0,%esp
- 804875b: ff 71 fc pushl -0x4(%ecx)
- 804875e: 55 push %ebp
- 804875f: 89 e5 mov %esp,%ebp
- 8048761: 56 push %esi
- 8048762: 53 push %ebx
- 8048763: 51 push %ecx
- 8048764: 83 ec 3c sub $0x3c,%esp //GCC 预留了0x3c个byte的空间用作main的局部变量
- A a1(10);
- 8048767: c7 44 24 04 0a 00 00 movl $0xa,0x4(%esp)
- 804876e: 00
- 804876f: 8d 45 dc lea -0x24(%ebp),%eax // a1就是ebp-24了
- 8048772: 89 04 24 mov %eax,(%esp)
- 8048775: e8 76 01 00 00 call 80488f0 <_ZN1AC1Ei>
- A a2(20);
- 804877a: c7 44 24 04 14 00 00 movl $0x14,0x4(%esp)
- 8048781: 00
- 8048782: 8d 45 d4 lea -0x2c(%ebp),%eax //a2就是 ebp-2c了
- 8048785: 89 04 24 mov %eax,(%esp)
- 8048788: e8 63 01 00 00 call 80488f0 <_ZN1AC1Ei>
- const A& ptr = a1 * a2;
- 804878d: 8d 45 cc lea -0x34(%ebp),%eax //这是GCC生成的内部匿名对象,作为参数传递了
- 8048790: 8d 55 d4 lea -0x2c(%ebp),%edx //a2
- 8048793: 89 54 24 08 mov %edx,0x8(%esp)
- 8048797: 8d 55 dc lea -0x24(%ebp),%edx //a1
- 804879a: 89 54 24 04 mov %edx,0x4(%esp)
- 804879e: 89 04 24 mov %eax,(%esp)
- 80487a1: e8 be 01 00 00 call 8048964 <_ZN1AmlERKS_>
- 80487a6: 83 ec 04 sub $0x4,%esp
- 80487a9: 8d 45 cc lea -0x34(%ebp),%eax
- 80487ac: 89 45 e4 mov %eax,-0x1c(%ebp)
- cout << "Fun out: [" << &ptr << "]" << endl;
- 然后看乘法的代码
- A operator*(const A& inA) { //重载“乘”, 函数以值返回一个对象
- 8048964: 55 push %ebp
- 8048965: 89 e5 mov %esp,%ebp
- 8048967: 57 push %edi
- 8048968: 56 push %esi
- 8048969: 53 push %ebx
- 804896a: 83 ec 2c sub $0x2c,%esp
- 804896d: 8b 5d 08 mov 0x8(%ebp),%ebx //现在的ebp是原来的esp(也就是老的EBP-0x3c,所以现在ebx指向的就是main里面那个匿名对象。
- A _ans;
- 8048970: 89 d8 mov %ebx,%eax 关键在这里,对局部变量进行构造的时候,GCC直接对main里面那个局部变量构造了。所以GCC省去了copy 构造这一步。
- 8048972: c7 44 24 04 00 00 00 movl $0x0,0x4(%esp)
- 8048979: 00
- 804897a: 89 04 24 mov %eax,(%esp)
- 804897d: e8 6e ff ff ff call 80488f0 <_ZN1AC1Ei>
复制代码 |
|