拷贝构造函数的参数类型(必须引用类型)

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

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

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


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


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

分享一道笔试题目,编译运行下图中的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. A(A other)
  11. {
  12. value = other.value;
  13. }
  14. void Print()
  15. {
  16. cout<<value<< endl;
  17. }
  18. };
  19. int main(void)
  20. {
  21. A a = 10;
  22. A b = a;
  23. b.Print();
  24. return 0;
  25. }
答案:编译错误。在复制构造函数中传入的参数是A的一个实例。由于是传值,把形参拷贝到实参会调用复制构造函数。因此如果允许复制构造函数传值,那么会形成永无休止的递归并造成栈溢出。因此C++的标准不允许复制构造函数传值参数,而必须是传引用或者常量引用。在Visual Studio和GCC中,都将编译出错。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
拷贝构造函数参数传递通常是通过引用来实现的。引用是一种特殊的变量类型,它允许我们使用已存在的变量作为参数传递给函数,而不是创建一个新的副本。在拷贝构造函数中,通常使用const引用作为参数类型,以确保被传递的对象不会被修改。这样可以避免不必要的内存开销和复制操作。 当以拷贝的方式初始化一个对象时,会调用一个特殊的构造函数,即拷贝构造函数。如果程序员没有显式地定义拷贝构造函数,编译器会自动生成一个默认的拷贝构造函数。这个默认的拷贝构造函数会使用已存在对象的成员变量来对新对象的成员变量进行一一赋值。换句话说,它会将已存在对象的属性值复制给新对象。这样,通过拷贝构造函数,我们可以在创建新对象时,传递一个已存在对象作为参数,从而实现对象的拷贝和初始化。 在实际编程中,我们也可以自定义拷贝构造函数,以便根据程序需求来实现更精确的拷贝操作。自定义的拷贝构造函数可以有不同的参数类型,但通常会使用const引用来传递被拷贝的对象。这样可以避免对被拷贝对象进行修改,同时也可以提高性能,避免不必要的复制和内存开销。 总结来说,拷贝构造函数参数传递通常使用const引用来实现,以确保被传递的对象不会被修改,同时也提高性能。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [C++拷贝构造函数(复制构造函数)详解](https://blog.csdn.net/ccc369639963/article/details/122905329)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值