返回值分析 

  函数不能通过返回指向栈内存的指针,返回指向堆内存的指针是可以的。

一、返回局部变量的值

可以有两种情况:返回局部自动变量和局部静态变量,比如:

int func()
{
int temp = ; // 返回局部自动变量的值,没问题
return temp;
}

局部变量temp存储在栈中,函数返回时会自动复制一份temp的copy给调用者,没有问题。

int func()
{
static int a = ; // 返回局部静态变量的值,没问题
return a;
}

局部变量a存储在静态(全局)存储区中,仅分配一次内存,并且函数返回后,变量不会销毁,没有问题。

vector<int> func()
{
vector<int> v;
v.push_back();
return v; //返回v的值拷贝,没问题
}
Person func()
{
Person p1;
p1.name = "test";
return p1; //返回值拷贝,会调用person类的拷贝构造函数,没问题
}

二、返回局部变量的指针

int* func()
{
int temp = ; //返回局部变量的地址
return &temp; //返回时讲已销毁的地址返回给调用者,结果不可预知
}
int* func()
{
static int temp = ;
return &temp; //返回指向静态存储区变量的指针, 没问题
}
char* func()
{
char *p = "test";
return p; //由于字符串test存储在常量存储区(不是静态存储区),返回一个指向常量的字符串指针是可行的。
}
char* func()
{
char str[] = "test";
return str; //函数返回一个已销毁的局部变量是不可行的。解决办法就是将字符串str声明为static。
}
char* func()
{
char *str = (char *)malloc(sizeof(char) * BUFFER_SIZE);
strcpy(str, "test");
return str; //函数返回一个指向堆内存的指针,由于堆存储区由程序员手动管理,可行的,但是要防止出现内存泄露,函数调用完后需要手动释放内存。
} char *temp = NULL;
temp = func();
// some operation...
free(temp);

三、返回局部变量的引用

int& func()
{
int temp = ; //返回局部变量的引用
return temp; //返回的temp本身,但已销毁,结果不可预知
}
 

四、静态全局变量和全局变量的区别

静态全局变量只在当前文件中可用,全局变量在其他文件中也可用,需要用extern声明。全局变量和静态变量如果没有手动初始化,则默认由编译器初始化为0。

#include <stdio.h>
char *returnStr()
{
char *p="hello world!"; //hello world!"是一个字符串常量,存放在只读数据段,把该字符串常量存放的只读数据段的首地址赋值给了指针
return p; //返回没问题
}
int main()
{
char *str;
str=returnStr();
printf("%s\n", str);
return ;
}
#include <stdio.h>
char *returnStr()
{
char p[]="hello world!"; //"hello world!"是局部变量存放在栈中。当returnStr函数退出时,栈要清空,局部变量的内存也被清空了
return p;
}
int main()
{
char *str;
str=returnStr();
printf("%s\n", str); //打印可能是乱码
return ;
}
int func()
{
int a;
....
return a; //允许
} int * func()
{
int a;
....
return &a; //无意义,不应该这样做
}

由于a返回的是值,无论是自动还是静态,返回局部变量的值是没有问题的;返回指向局部变量的指针,在调用结束后局部自动变量被销毁,指针指向一个不存在的对象,返回无意义。

#include <stdio.h>
char *returnStr()
{
static char p[]="hello world!"; //如果非要返回一个局部变量的地址,那么该局部变量一定要申明为static类型
return p;
}
int main()
{
char *str;
str=returnStr();
printf("%s\n", str);
return ;
}
char *GetMemory3(int num)
{
char *p = (char *)malloc(sizeof(char) * num);
return p; //返回指向堆内存的指针是可以的
}
void Test3(void)
{
char *str = NULL;
str = GetMemory3();
strcpy(str, "hello");
cout<< str << endl;
free(str);
}

c++ --> 返回值分析的更多相关文章

  1. linux recv函数返回值分析

    函数原型: ssize_t recv(int sockfd, void *buf, size_t len, int flags); 该函数第一个参数制定接收端套接字描述符; 第二个参数指明一个缓冲区, ...

  2. arm汇编进入C函数分析,C函数压栈,出栈,传参,返回值

    环境及代码介绍 环境和源码 由于有时候要透彻的理解C里面的一些细节问题,所有有必要看看汇编,首先这一切的开始就是从汇编代码进入C的main函数过程.这里不使用编译器自动生成的这部分汇编代码,因为编译器 ...

  3. 再分析 返回值加引用&,const

    本文主要分析,返回&,和返回值加const的作用. 返回& 定义一个数组模板: template<class T>class Array{ enum{size = 100} ...

  4. getchar()函数的返回值赋给char型,用if(ch=getchar() != EOF)测试,输入ctrl+z同样可以结束循环的分析

    2013-07-18 21:35:58 getchar()函数的返回值赋给char型,用if(ch=getchar() != EOF)测试,输入ctrl+z同样可以结束循环的分析. char是字符型数 ...

  5. springMVC源码分析--ViewNameMethodReturnValueHandler返回值处理器(三)

    之前两篇博客springMVC源码分析--HandlerMethodReturnValueHandler返回值解析器(一)和springMVC源码分析--HandlerMethodReturnValu ...

  6. springMVC源码分析--HandlerMethodReturnValueHandlerComposite返回值解析器集合(二)

    在上一篇博客springMVC源码分析--HandlerMethodReturnValueHandler返回值解析器(一)我们介绍了返回值解析器HandlerMethodReturnValueHand ...

  7. 分析RunTime执行命令以及得到返回值

    RunTime执行命令得到返回值 我们有在好好几篇博客里提到过RunTime,比如 JAVA之旅(二十三)--System,RunTime,Date,Calendar,Math的数学运算 Androi ...

  8. 字节码分析finally块对return返回值的影响

    直接进入主题.看如下代码: public int test(){ int i=0; try { i=1; return i; } catch (Exception e) { i=2; return i ...

  9. 源码分析springboot自定义jackson序列化,默认null值个性化处理返回值

    最近项目要实现一种需求,对于后端返回给前端的json格式的一种规范,不允许缺少字段和字段值都为null,所以琢磨了一下如何进行将springboot的Jackson序列化自定义一下,先看看如何实现,再 ...

随机推荐

  1. ActiveMQ的运用

    MQ的消息队列模式有2种,一种是点对点模式,一种是订阅模式. 点对点模式:点对点的模式主要建立在一个队列上面,当连接一个列队的时候,发送端不需要知道接收端是否正在接收,可以直接向ActiveMQ发送消 ...

  2. PHP simpleXML文件编程

    SimpleXML simpleXML该技术的核心思想是以面向对象的方式来操作xml文件 <?php //simplexml文件 讲所有的元素转成对象 $library=simplexml_lo ...

  3. 【mongodb系统学习之八】mongodb shell常用操作

    八.mongodb  shell常用基础操作(每个语句后可以加分号,也可以不加,看情况定(有的工具中可以不加),最好是加): 1).进入shell操作界面:mongo,上边已有演示: 2).查看当前使 ...

  4. CF368 D - Persistent Bookcase

    re了20多发 还是我在测试数据上操作最后了10多发才发现的 其实只需要多加一句就好了 真的愚蠢啊,要不都能进前100了 #include<bits/stdc++.h> using nam ...

  5. hdu2262 Where is the canteen

    Where is the canteen Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Ot ...

  6. 【Luogu3804】【模板】后缀自动机(后缀自动机)

    [Luogu3804][模板]后缀自动机(后缀自动机) 题面 洛谷 题解 一个串的出现次数等于\(right/endpos\)集合的大小 而这个集合的大小等于所有\(parent\)树上儿子的大小 这 ...

  7. 【BZOJ2440】完全平方数(二分答案,莫比乌斯反演)

    [BZOJ2440]完全平方数(二分答案,莫比乌斯反演) 题面 BZOJ 题解 很显然,二分一个答案 考虑如何求小于等于这个数的非完全平方数倍数的个数 这个明显可以直接,莫比乌斯反演一下 然后这题就很 ...

  8. BZOJ 1123:城市道路

    Sol 就是求割点,把贡献算一下就好...直接tarjan # include <bits/stdc++.h> # define RG register # define IL inlin ...

  9. linux系统基础优化16条知识汇总

    优化的总结: 1.不用root管理,以普通用户的名义通过sudo授权管理. 2.更改默认的远程连接SSH服务端口,禁止root用户远程连接,甚至 要更改只监听内网IP. 3.定时自动更新服务区时间,使 ...

  10. Cglib及其基本使用

    前言 最近一直在看Spring源码,其实我之前一直知道AOP的基本实现原理: 如果针对接口做代理默认使用的是JDK自带的Proxy+InvocationHandler 如果针对类做代理使用的是Cgli ...