【求助】关于取函数返回值地址的问题
遇到一个问题,请大侠们指点
以一个代码为例:
#include <iostream>
#include <string.h>
int getName()
{
int i = 20;
return i;
}
int main()
{
int *p;
p = &getName();
return 0;
}
对于基础类型,简单数据返回结果会返回在eax寄存器中,上面的代码会报错
但是如果返回的不是int而是一个对象的话,返回的效果应该是在栈上开一块空间传入到函数中,函数中把对象拷贝到对应空间处,返回返回值的指针在eax寄存器中。
但是如果我像下面这样的
ClassName getVal()
{
.....
}
int main()
{
ClassName *name;
name = &getVal();
...
return 0;
}
会得到一个编译器告警,编译成功,但是这个样子这块栈空间什么时候释放呢?是不是说因为我有一个指针的指向而一直不释放呢?怎么样才能体现他确实释放了呢? 我在name = &getVal();后写了一堆的调用函数的代码,函数的入参都很大,可是不管怎么多调用,最后取出name的值仍然是完好的,很迷茫啊,求解释
[解决办法]
内存在函数调用的表达式结束后就回收了.
回收并不代表内存就不可用了. 内存还在, 只是编译器随时可能把它用于其他用途.
如果你运气好, 编译刚好还没有把它用作其他的, 你就看不到错误.
[解决办法]
正解,我给代码解释下
咱先把引用去掉,在c下编译是通不过的
2 #include <stdio.h>
3 #include <string.h>
4 int getName()
5 {
6 int i = 20;
7 return i;
8 }
9 int main()
10 {
11 int *p;
12 *p = getName();
13 return 0;
14 }
gcc编译后,反汇编得到的结果
80483b4 <getName>:
119 80483b4: 55 push %ebp
120 80483b5: 89 e5 mov %esp,%ebp//
121 80483b7: 83 ec 10 sub $0x10,%esp
122 80483ba: c7 45 fc 14 00 00 00 movl $0x14,-0x4(%ebp)
123 80483c1: 8b 45 fc mov -0x4(%ebp),%eax//here1
/*很显然here1它把返回值放到了寄存器eax中,然后有主函数中here2存到对应的变量中去。
也就说这个操作执行后,函数getname中局部变量-0x4(%ebp),已经不再重要了,
因为它的值已经转移走了。系统可以用他干别的事情了。而且它是在栈里面的,局部于%ebp
在进入函数的时候push %ebp保存调用函数地址;mov %esp,%ebp转向被调用函数栈指针*/
124 80483c4: c9 leave
125 80483c5: c3 ret
126
127 080483c6 <main>:
128 80483c6: 55 push %ebp
129 80483c7: 89 e5 mov %esp,%ebp
130 80483c9: 83 ec 10 sub $0x10,%esp
131 80483cc: e8 e3 ff ff ff call 80483b4 <getName>
132 80483d1: 8b 55 fc mov -0x4(%ebp),%edx
133 80483d4: 89 02 mov %eax,(%edx)//here2
134 80483d6: b8 00 00 00 00 mov $0x0,%eax
135 80483db: c9 leave
136 80483dc: c3 ret
137 80483dd: 90 nop
138 80483de: 90 nop
139 80483df: 90 nop