细节----函数返回局部变量

先来看一个例子作为引入:

#include <stdio.h>
#include <stdlib.h>

int *test()
{
    int num = 3;
    return &num;
}

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 &num;
}

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 &num;
}

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. 字符常量存放在只读数据段中,所以可以返回地址。

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值