C++构造与析构(15) - 为何拷贝构造函数必须为const

当用户自定义拷贝构造函数时,通常传入的参数是const引用。

之所以使用const引用,其中一个原因是C++中当不想一个对象被意外修改时,则使用const来修饰。不过除此之外,还有别的原因。

例如,参考下面程序的结果:

注意:要确保编译器已经disable copy elision。否则编译还是可以通过。关闭方法可以参考本人之前的这篇文章

#include<iostream>
using namespace std;
 
class Test
{
public:
   Test(Test &t) { /*使用t来拷贝数据成员*/}
   Test()        { /*初始化数据成员*/ }
};
 
Test fun()
{
    cout << "fun() Called\n";
    Test t;
    return t;
}
 
int main()
{
    Test t1;
    Test t2 = fun();
    return 0;
}

root@shltsh:~$ g++ test.cpp -fno-elide-constructors
编译失败,提示:
Compiler Error in line "Test t2 = fun();" 

程序第一眼看上去是正常的,但是却编译失败了。

如果在拷贝构造函数中加上const,如下面这行代码所示,则编译正常。
    Test(const Test &t) { cout << "Copy Constructor Called\n"; }
或者,将这行代码
    Test t2 = fun();
修改为下面的这两行,程序也能编译正常。
    Test t2;
    t2 = fun();   //调用的赋值操作符

为什么这样?因为函数fun()是按值传递的,所以在最初的例子中,编译器创建了一个临时对象,并调用拷贝构造函数将其拷贝给t2。
编译失败的原因在于,编译器创建的这个临时对象,不能绑定给non-const的引用。因为修改一个编译器创建的临时对象是完全没有意义的,这个临时对象随时都会被析构掉。

  • 9
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 3
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值