先来看一个例子作为引入:
#include <stdio.h>
#include <stdlib.h>
int *test()
{
int num = 3;
return #
}
int main()
{
int *p = test();
printf("%d\n", *p);
return 0;
}
这样的一段代码,如果是gcc编译会有提示警告:函数返回局部变量的地址这样的warning,函数中返回变量的地址是错误的,这样的代码存在隐患,虽然这段代码运行的结果是正确的。
这是因为函数的调用会使用栈空间来存放函数中的局部变量,当函数调用完成之后,会释放占用的栈空间,所以返回地址实际上是没有任何意义的,返回的地址的空间马上就被其它函数占用。
看这样一段代码:
#include <stdio.h>
#include <stdlib.h>
int *test()
{
int num = 3;
return #
}
int test1()
{
int num1 = 4;
return num1;
}
int main()
{
int *p = test();
test1();
printf("%d\n", *p);
return 0;
}
运行这段代码,会惊奇的发现输出的结果是4而不是3,这是因为test1()的局部变量num1把原来*test()的局部变量num的地址空间占用了,但是这样的结果也是不可预料的,也有可能出现的是乱码,所以结论是我们不能在函数中返回局部变量的地址。不过我们可以通过加上static关键字让这个局部变量变成静态局部变量存在.data(已初始化)数据段中,还可以通过申请堆空间的方法。这样我们就可以返回它的地址了。
例子1(static):
#include <stdio.h>
#include <stdlib.h>
int *test()
{
static int num = 3;
return #
}
int test1()
{
int num = 4;
return num;
}
int main()
{
int *p = test();
test1();
printf("%d\n", *p);
return 0;
}
例子2(申请堆空间):
#include <stdio.h>
#include <stdlib.h>
int *test()
{
int *num = (int *)malloc(sizeof(int));
num[0] = 3;
return num;
}
int test1()
{
int num = 4;
return num;
}
int main()
{
int *p = test();
test1();
printf("%d\n", *p);
free(p);
return 0;
}
结果输出的是3,这样就达到了我们的目的。例子1中因为static int num根本就不存储在栈区中,而是数据区中,专门用于存放全局变量和局部静态变量,例子2中malloc申请的是堆空间的空间,除非自己手动free掉,或者程序运行结束后,才会回收。
还有,返回局部变量的值是允许的,因为只是一个赋值的行为,和地址无关。
最后,还有一个特例:
#include <stdio.h>
#include <stdlib.h>
char *test()
{
char *num = "123123";
return num;
}
char test1()
{
char num = '4';
return num;
}
int main()
{
char *p = test();
test1();
printf("%s\n", p);
return 0;
}
结果输出是正确的,这是因为字符常量存放在只读数据段中,该字符常量占用的空间并不会随着函数的退出而回收,而是程序运行结束之后回收。
总结一下:
1. 函数返回局部变量的地址是绝对禁止的,除非使用static修饰或者使用malloc或者new申请空间;
2. 返回局部变量的值是允许的;
3. 字符常量存放在只读数据段中,所以可以返回地址。
本文探讨了在C/C++中函数返回局部变量地址的问题,分析了潜在风险,并提供了使用static关键字和动态内存分配的解决方案。
900

被折叠的 条评论
为什么被折叠?



