在C++中, 构造函数,拷贝构造函数,析构函数和赋值函数(赋值运算符重载)是最基本不过的需要掌握的知识。 但是如果我问你“拷贝构造函数的参数为什么必须使用引用类型?”这个问题, 你会怎么回答? 或许你会回答为了减少一次内存拷贝? 很惭愧的是,我的第一感觉也是这么回答。不过还好,我思索一下以后,发现这个答案是不对的。

  原因
       如果拷贝构造函数中的参数不是一个引用,即形如CClass(const CClass c_class),那么就相当于采用了传值的方式(pass-by-value),而传值的方式会调用该类的拷贝构造函数,从而造成无穷递归地调用拷贝构造函数。因此拷贝构造函数的参数必须是一个引用。
  需要澄清的是,传指针其实也是传值,如果上面的拷贝构造函数写成CClass(const CClass* c_class),也是不行的。事实上,只有传引用不是传值外,其他所有的传递方式都是传值。

  先从一个小例子开始:(自己测试一下自己看看这个程序的输出是什么?)

 1 #include<iostream>
 2 using namespace std;
 3
 4 class CExample
 5 {
 6 private:
 7     int m_nTest;
 8
 9 public:
10     CExample(int x) : m_nTest(x)      //带参数构造函数
11     {
12         cout << "constructor with argument"<<endl;
13     }
14
15     // 拷贝构造函数,参数中的const不是严格必须的,但引用符号是必须的
16     CExample(const CExample & ex)     //拷贝构造函数
17     {
18         m_nTest = ex.m_nTest;
19         cout << "copy constructor"<<endl;
20     }
21
22     CExample& operator = (const CExample &ex)   //赋值函数(赋值运算符重载)
23     {
24         cout << "assignment operator"<<endl;
25         m_nTest = ex.m_nTest;
26         return *this;
27     }
28
29     void myTestFunc(CExample ex)
30     {
31     }
32 };
33
34 int main(void)
35 {
36     CExample aaa(2);
37     CExample bbb(3);
38     bbb = aaa;
39     CExample ccc = aaa;
40     bbb.myTestFunc(aaa);
41
42     return 0;
43 }  

  这个例子的输出结果是:

1 constructor with argument        // CExample aaa(2);
2 constructor with argument        // CExample bbb(3);
3 assignment operator              // bbb = aaa;
4 copy constructor                 // CExample ccc = aaa;
5 copy constructor                 //  bbb.myTestFunc(aaa);  

  如果你能一眼看出就是这个结果的话, 恭喜你,可以站起来扭扭屁股,不用再往下看了。

  如果你的结果和输出结果有误差, 那拜托你谦虚的看完。

  第一个输出: constructor with argument      // CExample aaa(2);

  如果你不理解的话, 找个人把你拖出去痛打一顿,然后嘴里还喊着“我是二师兄,我是二师兄.......”

  第二个输出:constructor with argument     // CExample bbb(3);

  分析同第一个

  第三个输出: assignment operator                // bbb = aaa;

  第四个输出: copy constructor                      // CExample ccc = aaa;

  这两个得放到一块说。 肯定会有人问为什么两个不一致。原因是, bbb对象已经实例化了,不需要构造,此时只是将aaa赋值给bbb,只会调用赋值函数,就这么简单,还不懂的话,撞墙去! 但是ccc还没有实例化,因此调用的是拷贝构造函数,构造出ccc,而不是赋值函数,还不懂的话,我撞墙去!!

  

  第五个输出: copy constructor                      //  bbb.myTestFunc(aaa);

  实际上是aaa作为参数传递给bbb.myTestFunc(CExample ex), 即CExample ex = aaa;和第四个一致的, 所以还是拷贝构造函数,而不是赋值函数, 如果仍然不懂, 我的头刚才已经流血了,不要再让我撞了,你就自己使劲的再撞一次吧。

  通过这个例子, 我们来分析一下为什么拷贝构造函数的参数只能使用引用类型。

  看第四个输出: copy constructor                      // CExample ccc = aaa;

  构造ccc,实质上是ccc.CExample(aaa); 我们假如拷贝构造函数参数不是引用类型的话, 那么将使得 ccc.CExample(aaa)变成aaa传值给ccc.CExample(CExample ex),即CExample ex = aaa,因为 ex 没有被初始化, 所以 CExample ex = aaa 继续调用拷贝构造函数,接下来的是构造ex,也就是 ex.CExample(aaa),必然又会有aaa传给CExample(CExample ex), 即 CExample ex = aaa;那么又会触发拷贝构造函数,就这下永远的递归下去。

  所以绕了那么大的弯子,就是想说明拷贝构造函数的参数使用引用类型不是为了减少一次内存拷贝, 而是避免拷贝构造函数无限制的递归下去

  附带说明,在下面几种情况下会调用拷贝构造函数

    a、   显式或隐式地用同类型的一个对象来初始化另外一个对象。如上例中,用对象c初始化d;
    b、  作为实参(argument)传递给一个函数。如CClass(const CClass c_class)中,就会调用CClass的拷贝构造函数;
    c、  在函数体内返回一个对象时,也会调用返回值类型的拷贝构造函数;
    d、  初始化序列容器中的元素时。比如 vector<string> svec(5),string的缺省构造函数和拷贝构造函数都会被调用;
    e、  用列表的方式初始化数组元素时。string a[] = {string(“hello”), string(“world”)}; 会调用string的拷贝构造函数。

  如果在没有显式声明构造函数的情况下,编译器都会为一个类合成一个缺省的构造函数。如果在一个类中声明了一个构造函数,那么就会阻止编译器为该类合成缺省的构造函数。和构造函数不同的是,即便定义了其他构造函数(但没有定义拷贝构造函数),编译器总是会为我们合成一个拷贝构造函数。 

  另外函数的返回值是不是引用也有很大的区别,返回的不是引用的时候,只是一个简单的对象,此时需要调用拷贝构造函数,否则,如果是引用的话就不需要调用拷贝构造函数。 

 1 #include<iostream>
 2 using namespace std;
 3
 4 class A
 5 {
 6 private:
 7     int m_nTest;
 8 public:
 9     A()
10     {
11     }
12     A(const A& other)    //构造函数重载
13     {
14         m_nTest = other.m_nTest;
15         cout << "copy constructor"<<endl;
16     }
17     A & operator =(const A& other)
18     {
19         if(this != &other)
20         {
21             m_nTest = other.m_nTest;
22             cout<<"Copy Assign"<<endl;
23         }
24         return *this;
25     }
26 };
27
28 A fun(A &x)
29 {
30     return x;     //返回的不是引用的时候,需要调用拷贝构造函数
31 }
32
33 int main(void)
34 {
35     A test;
36     fun(test);
37     system("pause");
38     return 0;
39 }  

  运行结果

  

  分享一道笔试题目,编译运行下图中的C++代码,结果是什么?(A)编译错误;(B)编译成功,运行时程序崩溃;(C)编译运行正常,输出10。请选择正确答案并分析原因。

 1 class A
 2 {
 3 private:
 4     int value;
 5 public:
 6     A(int n)
 7     {
 8         value = n;
 9     }
10
11     A(A other)
12     {
13         value = other.value;
14     }
15     void Print()
16     {
17         cout<<value<<endl;
18     }
19 };
20
21 int main(void)
22 {
23     A a = 10;
24     A b = a;
25     b.Print();
26     return 0;
27 }  

  答案:编译错误。在复制构造函数中传入的参数是A的一个实例。由于是传值,把形参拷贝到实参会调用复制构造函数。因此如果允许复制构造函数传值,那么会形成永无休止的递归并造成栈溢出。因此C++的标准不允许复制构造函数传值参数,而必须是传引用或者常量引用。在Visual Studio和GCC中,都将编译出错。

  

【转】拷贝构造函数的参数类型必须是引用相关推荐

  1. 拷贝构造函数的参数类型必须是引用

    在C++中, 构造函数,拷贝构造函数,析构函数和赋值函数(赋值运算符重载)是最基本不过的需要掌握的知识. 但是如果我问你"拷贝构造函数的参数为什么必须使用引用类型?"这个问题, 你 ...

  2. C++拷贝构造函数的参数为什么必须使用引用类型(无限递归)

    在C++中, 构造函数,拷贝构造函数,析构函数和赋值函数(赋值运算符重载)是最基本不过的需要掌握的知识. 但是如果我问你"拷贝构造函数的参数为什么必须使用引用类型?"这个问题, 你 ...

  3. Cpp / 拷贝构造函数的参数为什么必须使用引用类型

    表面原因:编译器无法通过,会报如下错误: error: invalid constructor; you probably meant 'CExample (const CExample&)' ...

  4. C++中拷贝构造函数的形参为什么要是const引用

    1.首先为什么要是引用: 如对于 CBox(CBox initB);//复制构造函数一开始想到的原型 CBox cigar; CBox myBox(cigar); //如果编写这样一条语句 //那么将 ...

  5. 拷贝构造函数 浅拷贝与深拷贝

    目录 拷贝构造函数 浅拷贝 深拷贝 拷贝构造函数 上一期中我们讲述了构造函数的相关内容,谈到构造函数在形式上有几种分类,即带参数的.不带参数的以及参数列表初始化的,还有一种传引用的构造函数,称为拷贝构 ...

  6. C++拷贝构造函数与拷贝赋值运算符

    拷贝构造函数 拷贝构造函数定义了当用同类型的另一个对象初始化新对象时做什么,如果一个构造函数的第一个参数是自身类类型的引用,且任何额外参数都有默认值,则此构造函数是拷贝构造函数.拷贝构造函数的参数类型 ...

  7. C++ 拷贝构造函数和赋值构造函数

    在C++中复制控制是一个比较重要的话题,主要包括复制构造函数.重载赋值操作符.析构函数这三部分,这三个函数是一致的,如果类需要析构函数,则它也需要复制操作符 和 复制构造函数,这个规则被称为 C++的 ...

  8. 默认拷贝构造函数 与 自定义拷贝构造函数

    拷贝构造 有Person类如下: class Person {char* pName; public:Person(char* pN="noName"){cout<<& ...

  9. 拷贝构造函数的第一个参数必须是自身类类型的引用

    拷贝构造函数的第一个参数必须是自身类类型的引用 拷贝构造函数的定义:如果一个构造函数的第一个参数是自身类类型的引用,且任何额外参数都有默认值,则此构造函数是拷贝构造函数 第一个参数必须是自身类类型的引 ...

最新文章

  1. Keras情感分析(Sentiment Analysis)实战---自然语言处理技术
  2. vs2012下 error4996
  3. rsa加密算法python_模拟新浪微博登录(Python+RSA加密算法)
  4. redis 简单应用
  5. (带后台)苹果安卓APP下载页 app应用下载界面源码
  6. 不会写代码也能当程序员?无代码来了,是福还是祸?
  7. 狡兔死,良弓藏-每个王朝都一样
  8. 研发体系核心代码和文档安全保护方案
  9. 嵌入式Linux系统驱动hp1020打印机
  10. Python:2行代码实现文字转语音
  11. dplayer安装php_Dplayer播放器集成p2p加速源码分享
  12. 单例模式详解(线程安全,饿汉,懒汉模式)
  13. 5种经典的数据分析思维和方法
  14. 苹果可以访问linux的smb,Samba For iOS让iPhone使用网上邻居使用体验教程
  15. 51单片机的温湿度控制系统
  16. 识别滑块缺口位置DLL
  17. 中国废弃资源市场现状趋势与发展方向分析报告2022-2027年
  18. 《精彩网址大全——生活资讯文体娱乐卷》前言
  19. trim()方法的使用
  20. iHost 域名解析

热门文章

  1. java 创建restful_使用Java创建RESTful Web Service
  2. java p8级别_JAVA程序员月入5000+很迷茫,如何能在一年内改变达到月入过万?
  3. python knnmatch_opencv python 特征匹配
  4. 使命召唤ol服务器位置,服务器架构升级 使命召唤OL跨区作战时代来临!
  5. mybatis 关联查询 延迟加载
  6. C++ 构造函数和析构函数
  7. java高级必须懂得_反射---Java高级开发必须懂的
  8. 计算机系统结构自学试卷,全国2004年4月高等教育自学考试计算机系统结构试题...
  9. 张一鸣批评的互联网“语言腐败”,危害到底有多大?
  10. vSphere Datacenter设计– vSphere 6.0中的vCenter体系结构更改–第2部分