免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2462 | 回复: 5
打印 上一主题 下一主题

[C++] 请教一个关于返回引用的问题 [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2011-07-19 10:34 |只看该作者 |倒序浏览
本帖最后由 deewenic 于 2011-07-19 10:36 编辑

有如下代码:
  1. #include <iostream>
  2. using namespace std;

  3. class rational
  4. {
  5.     private:
  6.         int real;
  7.         int image;
  8.     public:
  9.         rational(){real = 0; image = 0;};
  10.         rational(int r = 0, int i = 0);
  11.         ~rational(){cout<<"dctor"<<endl;}
  12.         rational& operator*(rational& rhs);
  13.         void showval();
  14. };

  15. void rational::showval(){
  16.     cout << "The value is ("
  17.          << real
  18.          << ","
  19.          << image
  20.          << ")" << endl;
  21. }

  22. rational::rational(int r,int i)
  23. {
  24.      cout<<"rational(" << r << "," << i << ")"<<endl;
  25.      real = r;
  26.      image = i;
  27. }
  28. /*****①:返回局部对象**********************/
  29. rational& rational::operator*(rational& rhs)
  30. {
  31.      cout<<"in operator*"<<endl;
  32.      rational ret(real*rhs.real,image*rhs.image);
  33.      return ret;
  34. }
  35. /*****②:返回指针指向的对象***************/
  36. rational& rational::operator*(rational& rhs)
  37. {
  38.      rational* result = new rational(real*rhs.real, image*rhs.image);
  39.      return *result;
  40. }

  41. int main()
  42. {
  43.      rational a(1,2);
  44.      rational b(3,4);
  45.      rational c(2,2);

  46.     rational ret = a * b * c;
  47.     ret.showval();
  48.      return 0;
  49. }
复制代码
问题:
当我使用第一个方法重载乘号操作时,在SunOS下用CC编译会报Warning:Returning a reference to a local variable or temporary.
当我使用第二个方法重载乘号操作时,同样环境下编译正常。

就我的理解:因为第一个返回局部对象,而这个对象是放在栈上的,函数返回的时候这个对象应该会被释放。而第二个方法返回的是在堆上的对象,所以不会被释放,因此返回可用。问题的关键在于,在两种方法都编译过后并执行,执行结果即算出的有理数值是一样的,这说明返回局部对象是正确的么?或者说如果返回局部对象会有什么弊端?请大家帮忙分析一下两者的异同,谢谢了。

论坛徽章:
0
2 [报告]
发表于 2011-07-19 10:39 |只看该作者
第二种方法,你只返回了指针,编译器无法判断你指针的值是否有问题

论坛徽章:
0
3 [报告]
发表于 2011-07-19 10:57 |只看该作者
首先,你使用第二种方式返回一个new出来的对象的引用,过程中使用过一次默认的拷贝构造函数,可是你的类里没有定义和实现,存在隐患。
第二,使用第一种方式返回的栈变量的引用,虽然在忍受编译警告的情况下仍然可以编译通过并运行,但是它是很不安全的使用栈变量的方式,假如在你调用ret.showval()这个函数之前,你再调用别的函数fun,假设你定义了一个别的成员函数为:void rational::fun(){ int val[200] = {0,};memset(val, 0, 200);},当调用这个函数之后,栈内原本存放rational引用的那块内存被覆写,再调用showval()之后你就会发现有问题了。

以上是理念流,你可以自己实践一下:)

good luck

论坛徽章:
1
15-16赛季CBA联赛之四川
日期:2016-02-15 17:12:32
4 [报告]
发表于 2011-07-19 12:33 |只看该作者
首先,你使用第二种方式返回一个new出来的对象的引用,过程中使用过一次默认的拷贝构造函数,可是你的类里没 ...
foolishx 发表于 2011-07-19 10:57



    返回引用不会去调用拷贝构造的

论坛徽章:
0
5 [报告]
发表于 2011-07-19 12:54 |只看该作者
the second is right metod;
though the first metod return the value is what you expect , it is not reliable!
becuase the local variable you returned is a reference which references to a local variable, that is to say it points to a local; at the moment the OS hasn't retrieved the local value, so you got the right value

论坛徽章:
0
6 [报告]
发表于 2011-07-19 12:57 |只看该作者
in addition, returning reference doesn't need to call the copy constructor function.
4 floor said right
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP