C++函数返回值的特殊情况(返回引用、非引用、引用左值)

关于函数返回值的问题,关键是我们要清楚返回过程中,值得复制过程

1、返回内置类型

      string make_plural(size_t i,const string &word,const string &ending)
      {
             return (i==1)?word:word+ending;
      }

      以上函数,当i等于1时,函数返回word形参的副本;当i不等于1时函数返回一个临时的string对象,这个临时对象是由字符串word和ending相加而成的。

        这两种情况下,return都在调用该函数的地方复制了返回的string对象

      (首先,func将返回值复制到一个匿名临时变量中,在这里假设该临时变量为anony(其实是没有名字的,这里方便阐述);然后,再将anony的值复制到result,可以看出,这里是进行了两次复制的。而并非一次复制。

      (一个比较大的缺点就是如果返回值比较大的话,函数的性能会受到一定的影响


2、返回指针


。。。



2、返回引用

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

      以上函数的返回值是引用类型。无论返回s1或是s2,调用函数和返回结果时,都没有复制这些string对象

        千万不要返回局部对象的引用

      const string &mainip(const string &s)
      {
             string ret=s;
             return ret;
      }
      
      当函数执行完毕,程序将释放分配给局部对象的存储空间。此时,对局部对象的引用就会指向不确定的内存。(我在dev c++里运行通过。。。。 但是vc6和vc2008不能编译通过)。同理,也不能返回局部对象的指针。


最后补一个例子:


   

// LJJ.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

#include <string>
#include <iostream>
using namespace std;

const int &weatherb(const int &b){
	return b;
}

int main(int argc, char* argv[])
{
	int b = 1;
	int result ;	
	
	cout<<"b:"<<&b<<endl;
	cout<<"result:"<<&result<<endl;

	result= weatherb(b);
	cout<<endl<<"b:"<<&b<<" "<<b<<endl;
	cout<<"result:"<<&result<<" "<<result<<endl;
	
	b+=1;
	result+=2;
	cout<<endl<<"b:"<<&b<<" "<<b<<endl;
	cout<<"result:"<<&result<<" "<<result<<endl;

	return 0;
}

运行结果:

   

需要说明的是:按很多人的理解,这里返回的是一个引用,因此result就是b的引用,其实并非如此,这里返回引用只是减少了一次临时变量值的复制。如果真的要让result能够引用b,可以这样做:int &result = func(b);

3、引用返回左值(比较奇妙)

      char &get_val(string &str,string::size_type ix)
      {
             return str[ix];
      }

      使用语句调用:
       string s("123456");
       cout<<s<<endl;
       get_val(s,0)='a';
       cout<<s<<endl;
      把函数应用于左值,还是第一次见到(真是孤陋寡闻)。


// LJJ.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"

#include <string>
#include <iostream>
using namespace std;

int &re(int &b){
	return b;
}


int main(int argc, char* argv[])
{
	int b = 1;
	cout<<b<<endl;//1
	re(b) = 2;
	cout<<b<<endl;//2
	return 0;
}



以下是上面几种情况的实例代码:

// liujiajia
// 2008-7-16
#include < iostream >
using   namespace  std;
string  make_plural(size_t, const   string & , const   string & );
const   string   & shorterString( const   string   & , const   string   & );
const   string   & mainip( const   string & );
char   & get_val( string   & , string ::size_type);
int  main( void )
{
    cout
<< make_plural( 1 , " dog " , " s " ) << endl;
    cout
<< make_plural( 2 , " dog " , " s " ) << endl;
    
    
string  string1 = " 1234 " ;
    
string  string2 = " abc " ;
    cout
<< shorterString(string1,string2) << endl;
    
    cout
<< mainip( " jiajia " ) << endl;
    
    
    
string  s( " 123456 " );
    cout
<< s << endl;
    get_val(s,
0 ) = ' a ' ;
    
    cout
<< s << endl;
    
    getchar();
    
return   0 ;
}
// 返回非引用 
string  make_plural(size_t i, const   string   & word, const   string   & ending)
{
    
return  (i == 1 ) ? word:word + ending;
}
// 返回引用 
const   string   & shorterString( const   string   & s1, const   string   & s2)
{
    
return  s1.size() < s2.size() ? s1:s2;
}
// 禁止返回局部对象的引用(我的dev c++ 没有报错,比较可怕) 
const   string   & mainip( const   string   & s)
{
    
string  ret = s;
    
return  ret;
}
// 引用返回左值(第一次听说) 
char   & get_val( string   & str, string ::size_type ix)
{
    
return  str[ix];
}

from : http://www.cppblog.com/liujiajia/archive/2008/07/16/56244.html


刚又有一个例子,关于const的:

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

// d1.cpp : Defines the entry point for the console application.
//

#include "stdafx.h"
#include <iostream>
using namespace std;

const int &abc(int a,int b,int c,int &re){
re = a+b+c;
return re;
}

int main(int argc, char* argv[])
{
	int a = 1;
	int b = 2;
	int c = 3;
	int z;
	//abc(a,b,c,z)++;			//<-------------error C2166: l-value specifies const object
	abc(a,b,c,z);				//right!!
	z++;						//right!!
	cout<<z<<endl;				//7
	return 0;
}



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值