一,c++函数的返回分为以下几种情况

1)主函数main的返回值:这里提及一点,返回0表示程序运行成功。

2)返回非引用类型:函数的返回值用于初始化在跳用函数出创建的临时对象。用函数返回值初始化临时对象与用实参初始化形参的方法是一样 的。如果返回类型不是引用,在调用函数的地方会将函数返回值复制给临时对象。且其返回值既可以是局部对象,也可以是求解表达式的结果。

3)返回引用:当函数返回引用类型时,没有复制返回值。相反,返回的是对象本身。

二,函数返回引用

1,当函数返回引用类型时,没有复制返回值。相反,返回的是对象本身。先看两示例,示例1如下:

const string &shorterString(const string &s1,const string &s2)
{
return s1.size < s2.size ? s1:s2;
}

示例2:

  1. ostream &operator<<(ostream &output, const AAA &aaa)
  2. {
  3. output << aaa.x << ' ' << aaa.y << ' ' << aaa.z << endl;
  4. return output;
  5. }

形参和返回类型都是指向const string对象的引用,调用函数和返回结果时,都没有复制这些string对象。

2,返回引用,要求在函数的参数中,包含有以引用方式或指针方式存在的,需要被返回的参数。比如:

int& abc(int a, int b, int c, int& result){

result = a + b + c;
    return result;
}

这种形式也可改写为:

int& abc(int a, int b, int c, int *result){
   *result = a + b + c;
   return *result;
}

但是,如下的形式是不可以的:
int& abc(int a, int b, int c){
   return a + b + c;
}

3,千万不要返回局部对象的引用。当函数执行完毕时,将释放分配给局部对象的存储空间。此时,对局部对象的引用就会指向不确定的内存。如:

const string &manip(const string &s)

{

  string ret =s;

  return ret;  //wrong:returning reference to a local object

}

4,引用返回左值。返回引用的函数返回一个左值。因此这样的函数可用于任何要求使用左值的地方。示例见:c++ primer p215

5,由于返回值直接指向了一个生命期尚未结束的变量,因此,对于函数返回值(或者称为函数结果)本身的任何操作,都在实际上,是对那个变量的操作,这就是引入const类型的返回的意义。当使用了const关键字后,即意味着函数的返回值不能立即得到修改!如下代码,将无法编译通过,这就是因为返回值立即进行了++操作(相当于对变量z进行了++操作),而这对于该函数而言,是不允许的。如果去掉const,再行编译,则可以获得通过,并且打印形成z = 7的结果。

include <iostream>
include <cstdlib>
const int& abc(int a, int b, int c, int& result){
   result = a + b + c;
   return result;
}

int main() {
   int a = 1; int b = 2; int c=3;
   int z;
   abc(a, b, c, z)++;  //wrong: returning a const reference
   cout << "z= " << z << endl;
  SYSTEM("PAUSE");
   return 0;
}

三,思考:

1,什么时候返回引用是正确的?而什么时候返回const引用是正确的?

返回指向函数调用前就已经存在的对象的引用是正确的。当不希望返回的对象被修改时,返回const引用是正确的。

 MYTEST1:

本机测试,函数返回问题。 发现在mac上xcode中,返回直接返回局部变量,并没有拷贝,而是直接替换。

//
// main.cpp
// TestVector
//
// Created by New_Life on 2017/4/19.
// Copyright © 2017年 chenhuan001. All rights reserved.
// #include <iostream> class A {
public:
A() {
std::cout << "A construct" << std::endl;
} ~A() {
std::cout << "A destory" << std::endl;
} A(const A& a) {
std::cout << "A copy" << std::endl;
} A& operator = (const A& a) {
std::cout << "A =" << std::endl;
return *this;
}
int aa;
}; A Test() {
A a;
std::cout << "a: " << &a << std::endl;
A b(a);
std::cout << "b: " << &b << std::endl;
return b;
} int main(int argc, const char * argv[]) {
A c = Test();//编译器做了优化。
// A c(TestVector());
std::cout << "c: " << &c << std::endl;
A d;
std::cout << "d: " << &d << std::endl;
//std::cout << c << std::endl;
return ;
}

运行结果:

A construct
a: 0x7fff5fbff6b8
A copy
b: 0x7fff5fbff738
A destory
c: 0x7fff5fbff738
A construct
d: 0x7fff5fbff728
A destory
A destory

从结果中可以看出,b和c的地址相同。

并且在函数内的局部变量b,并没有析构。

(猜想,编译器的优化)

MYTEST2:

//
// main.cpp
// TestVector
//
// Created by New_Life on 2017/4/19.
// Copyright © 2017年 chenhuan001. All rights reserved.
// #include <iostream> class A {
public:
A() {
std::cout << "A construct" << std::endl;
} ~A() {
std::cout << "A destory" << std::endl;
} A(const A& a) {
std::cout << "A copy" << std::endl;
} A& operator = (const A& a) {
std::cout << "A =" << std::endl;
return *this;
}
int aa;
}; A& TestVector() {
A a;
std::cout << "a: " << &a << std::endl;
A b(a);
std::cout << "b: " << &b << std::endl;
return b;
} int main(int argc, const char * argv[]) {
A& c = TestVector();//编译器做了优化。
// A c(TestVector());
std::cout << "c: " << &c << std::endl;
A d;
std::cout << "d: " << &d << std::endl;
//std::cout << c << std::endl;
return ;
}

结果:

A construct
a: 0x7fff5fbff6b8
A copy
b: 0x7fff5fbff6a8
A destory
A destory
c: 0x7fff5fbff6a8
A construct
d: 0x7fff5fbff730
A destory

这次实验,函数返回了引用,可以发现c指向了未知内存。

c++ 函数返回研究[转]的更多相关文章

  1. Python从线程获取函数返回值

    Python中利用强大的threading模块可以很容易的实现多线程开发,提高运行速度.这一般是对某个进行大量计算操作的的函数进行多线程处理,然后合并各线程的结果.获取函数返回值的方法可以如下: 1) ...

  2. C++函数返回局部指针变量

    遇到过好几次关于函数返回指针变量问题,有时候是可以的,有时候是不可以的,然后就混乱了.今天研究了下,结果发现原来和内存分配有关. 用下面的例子分析下吧: char * test() { char a[ ...

  3. Linux下tcp协议socket的recv函数返回时机分析(粘包)

    http://www.vckbase.com/index.php/wv/10http://blog.csdn.net/zlzlei/article/details/7689409 文章一: 当前在网络 ...

  4. 由JS函数返回值引发的一场”血案"

    ---恢复内容开始--- 啊...  本来昨天晚上想写来着,结果陪老婆看电视剧就忘了... 呢滴神啊,原谅我吧. 背景:昨天在项目中做一个小功能的时候,出现了个小问题,而且一开始找了半天也没找到原因. ...

  5. 用javascript 写个函数返回一个页面里共使用了多少种HTML 标签

    今天我无意间看到一个面试题: 如何用javascript 写个函数返回一个页面里共使用了多少种HTML 标签? 不知你看到 是否蒙B了,如果是我 面试,肯定脑子嗡嗡的响.... 网上搜了搜也没有找到答 ...

  6. Entity Framework 6 Recipes 2nd Edition(11-1)译 -> 从“模型定义”函数返回一个标量值

    第11章函数 函数提供了一个有力代码复用机制, 并且让你的代码保持简洁和易懂. 它们同样也是EF运行时能利用的数据库层代码.函数有几类: Rowset Functions, 聚合函数, Ranking ...

  7. Entity Framework 6 Recipes 2nd Edition(11-2)译 -> 为一个”模型定义”函数返回一个计算列

    11-3. 为一个”模型定义”函数返回一个计算列 问题 想从”模型定义”函数里返回一个计算列 解决方案 假设我们有一个员工(Employee)实体,属性有: FirstName, LastName,和 ...

  8. Entity Framework 6 Recipes 2nd Edition(11-5)译 -> 从”模型定义”函数返回一个匿名类型

    11-5. 从”模型定义”函数返回一个匿名类型 问题 想创建一个返回一个匿名类型的”模型定义”函数 解决方案 假设已有游客(Visitor) 预订(reservation)房间(hotel ) 的模型 ...

  9. 函数返回多个值(c/c++)

    当我们在处理一个数组的时候常常会碰到这样的问题:输入一个数组,和数组的一个元素,返回该元素所在行数和列数.这样就需要返回多组两个值,且组数不定.上述类型的函数在c语言程序里面存在两个问题.第一,函数只 ...

随机推荐

  1. jQuery 选择器大全总结

    jQuery基础语法中规定的选择器有三种,分别是类选择器.ID选择器.标签选择器.如:$(“.aa”).$(“#id”).$(“div”),但中实际的应用中,DOM机构非常复杂,层级非常多.如和应对这 ...

  2. Android—— Intent参数this问题

    Android Intent参数this问题 (2013-04-02 11:19:48) 转载▼ 标签: android intent 分类: Android 转自:http://blog.csdn. ...

  3. KBEngine 服务器端-loginapp-协议构建、解析执行详细介绍

    宏宏宏 由于 C++ 是静态语言,不能像 js 一样通过函数名字符串来直接执行函数,所以将 messageId 映射到可执行函数的复杂性大大提升:KBEngine 使用了一系列精巧的「宏」来解决这个问 ...

  4. ElasticSearch使用代码

    package elasticsearch01; import static org.junit.Assert.*; import java.util.HashMap; import java.uti ...

  5. 关于Jsp中的三种弹框

    对话框有三种 1:只是提醒,不能对脚本产生任何改变: 2:一般用于确认,返回 true 或者 false ,所以可以轻松用于 if...else...判断 3:一个带输入的对话框,可以返回用户填入的字 ...

  6. zebra/quagga线程分析

    /* 线程按照不同的功能进行分类.有6条双链,分别表示不同类型的线程.将要运行的时候, * 就从不同的链表中取出,添加到ready链表中,运行完成之后,将线程结构体清空放到 * unuse链表中.一般 ...

  7. Ubuntu下启动/重启/停止apache服务器

    Task: Start Apache 2 Server /启动apache服务# /etc/init.d/apache2 startor$ sudo /etc/init.d/apache2 start ...

  8. Wellner 自适应阈值二值化算法

    参考文档: Adaptive Thresholding for the DigitalDesk.pdf       Adaptive Thresholding Using the Integral I ...

  9. linux -- Ubuntu图形界面终端实现注销、关机、重启

    linux命令行下的开关机分别为: 注销:logout 关机:shutdown -h now (参数不同,会有不同的关机模式) 重启:shutdown -r now 或 reboot 这些命令都是在l ...

  10. [mysql] 先按某字段分组再取每组中前N条记录

    From: http://blog.chinaunix.net/uid-26729093-id-4294287.html 请参考:http://bbs.csdn.net/topics/33002126 ...