• C++ 类和对象篇(六) 拷贝构造函数


    目录

    一、 概念

    1. 拷贝构造函数是什么?

    2. 为什么要有拷贝构造函数?

    3. 怎么用拷贝构造函数?

    3.1 创建拷贝构造函数

    3.2 调用拷贝构造函数

    二、特征

    三、合成拷贝构造函数

    1. 是什么?

    2. “双重删除”问题

    3. 什么时候需要显示的写拷贝构造函数?

    四、在拷贝构造函数中实现深拷贝

    1. 自己开辟一个新空间,然后将内容拷贝到新空间。

    2. 借助构造函数来实现深拷贝。

    【总结】

    【源代码】 


    一、 概念

    1. 拷贝构造函数是什么?

           拷贝构造函数是一个特殊的构造函数,也是用来初始化对象的,不过它是用已经存在的对象来初始化同类对象。

    2. 为什么要有拷贝构造函数?

           在创建新对象时,可否用已经存在的同类对象来初始化这个新对象呢?能否快速拷贝出一个对象的副本呢?

           为解决以上问题,C++中引入了拷贝构造函数:拷贝构造函数用于实现对象的复制和初始化。

    3. 怎么用拷贝构造函数?

    3.1 创建拷贝构造函数

           和构造函数一样,函数名和类名相同,且没有返回值,但拷贝构造函数的参数是当前类类型对象的引用或是指向当前类类型对象的指针。不能用当前类类型对象作为参数,这样会引发无穷递归问题。

    1. class Date
    2. {
    3. public:
    4. Date(int year = 2022, int month = 10, int day = 1)
    5. {
    6. _year = year;
    7. _month = month;
    8. _day = day;
    9. }
    10. //Date(const Date d) // 错误写法:编译报错,会引发无穷递归。
    11. // 正确写法,拷贝构造函数的参数必须是本类类型的引用或是指向本类类型的指针。
    12. Date(const Date& d)
    13. {
    14. _year = d._year;
    15. _month = d._month;
    16. _day = d._day;
    17. }
    18. Date(const Date* d)
    19. {
    20. _year = d->_year;
    21. _month = d->_month;
    22. _day = d->_day;
    23. }
    24. // 拷贝构造函数可以有多个参数,但一般只以一个对象为副本进行拷贝,所以只写一个参数。
    25. Date(const Date& d1,const Date& d2,const Date* d3)
    26. {
    27. _year = d1._year;
    28. _month = d2._month;
    29. _day = d3->_day;
    30. }
    31. private:
    32. int _year;
    33. int _month;
    34. int _day;
    35. };

            所以拷贝构造函数的参数必须是当前类类型对象的引用或是指向本类类型的指针,不然会引起以下拷贝构造函数传值传参带来的无穷递归问题

            拷贝构造函数如使用传值传参的方式,会引发无穷递归调用编译器会直接报错。因为使用传值传参时,编译器要调用拷贝构造函数用实参初始化形参,使用拷贝构造函数就必须得先传参,又会调用拷贝构造函数,这样就造成了造成无穷递归。


            在传引用传参或指针传参时,对于像拷贝构造函数的形参这类不需要修改的参数,建议加上一个const。这样做即能避免误操作导致实参被修改,也能误操作时给我们一个提示,让我们快速定位错误。

    eg. 加上const后实参不能被修改。

    3.2 调用拷贝构造函数

    a. 参数是 对象的引用 的拷贝构造函数

    1. int main()
    2. {
    3. Date a(1,1,1);
    4. Date b(a);
    5. return 0;
    6. }


    b. 参数是 指向对象的指针 的拷贝构造函数

    1. int main()
    2. {
    3. Date c(3, 3, 3);
    4. Date d(&c);
    5. return 0;
    6. }


    c. 多个参数的拷贝构造函数

    拷贝构造函数可以有多个参数,但一般只以一个对象为副本进行拷贝只写一个参数。

    1. int main()
    2. {
    3. Date a(1, 1, 1);
    4. Date b(2, 2, 2);
    5. Date c(3, 3, 3);
    6. Date d(a, b, &c);
    7. return 0;
    8. }


    二、特征

    拷贝构造函数也是特殊的成员函数,其特征如下:

    1. 拷贝构造函数是构造函数的一个重载形式。

    2. 拷贝构造函数的参数必须是本类类型的引用或是指向本类类型的指针。使用传值传参方式编译器会直接报错,因为会引发无穷递归调用。

    3. 拷贝构造函数可以有多个参数,但一般只以一个对象为副本进行拷贝,所以只写一个参数。

    4. 若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。


    三、合成拷贝构造函数

    1. 是什么?

    若未显式定义,编译器会生成默认的拷贝构造函数,叫做合成拷贝构造函数。

    注意:在编译器生成的默认拷贝构造函数中,内置类型是按照字节方式直接拷贝的,是浅拷贝,而自定义类型是调用其拷贝构造函数完成拷贝的。

    2. “双重删除”问题

            编译器生成的默认拷贝构造函数只能进行浅拷贝,无法对申请的资源(如动态开辟的空间)进行拷贝,看下面的例子:

    1. typedef int DataType;
    2. class Stack
    3. {
    4. public:
    5. Stack(size_t capacity = 10)
    6. {
    7. _array = (DataType*)malloc(capacity * sizeof(DataType));
    8. if (nullptr == _array)
    9. {
    10. perror("malloc申请空间失败");
    11. return;
    12. }
    13. _size = 0;
    14. _capacity = capacity;
    15. }
    16. ~Stack()
    17. {
    18. if (_array)
    19. {
    20. free(_array);
    21. _array = nullptr;
    22. _capacity = 0;
    23. _size = 0;
    24. }
    25. }
    26. private:
    27. DataType* _array;
    28. size_t _size;
    29. size_t _capacity;
    30. };
    31. int main()
    32. {
    33. Stack s1;
    34. s1.Push(1);
    35. s1.Push(2);
    36. s1.Push(3);
    37. s1.Push(4);
    38. Stack s2(s1); //调用默认拷贝构造函数
    39. return 0;
    40. }


            在以上例子中我们发现,默认的拷贝构造函数会令拷贝的类和被拷贝的类中的指针变量指向同一块空间,这样会造成同一块空间被析构函数析构两次,这通常被称为“双重删除”或“重复删除”,这是一个严重的问题,会导致程序崩溃。这个时候需要显示的写一个构造函数,并在里面完成深拷贝

    3. 什么时候需要显示的写拷贝构造函数?

            编译器生成的默认拷贝构造函数只能进行浅拷贝,无法对申请的资源(如动态开辟的空间)进行拷贝。

            所以类中如果没有涉及资源申请时,拷贝构造函数是否写都可以。一旦涉及到资源申请时,一定要显示的写拷贝构造函数,否则就是浅拷贝,可能导致“双重删除”问题。


    四、在拷贝构造函数中实现深拷贝

    1. 自己开辟一个新空间,然后将内容拷贝到新空间

    2. 借助构造函数来实现深拷贝。

           在构造函数中自然要动态成员变量开辟空间,所以在拷贝构造函数中可以使用构造函数创建一个临时对象,然后交换对象和临时对象的动态成员


           但这有个小问题,就是当前对象未初始化,直接交换数值可能会导致程序崩溃,所以加上初始化列表,在交换数值前先初始化。

    最后再给new的失败加一个提示或抛异常。

    想深入了解C/C++中深浅拷贝问题的同学,不妨看看博主的这篇文章: 「C/C++ 01」 深拷贝和浅拷贝


    【总结】


    【源代码】 

    1. //class A
    2. //{
    3. //public:
    4. // int a;
    5. // A(int _a) { a = _a; }
    6. //};
    7. //class Date
    8. //{
    9. //public:
    10. // Date(int year = 2022, int month = 10, int day = 1)
    11. // {
    12. // _year = year;
    13. // _month = month;
    14. // _day = day;
    15. // }
    16. // //Date(Date d) // 错误写法:编译报错,会引发无穷递归
    17. // // 正确写法,拷贝构造函数的参数必须是本类类型的引用或是指向本类类型的指针。
    18. // Date(const A& a)
    19. // {
    20. // _year = a.a;
    21. // _month = a.a;
    22. // _day = a.a;
    23. // }
    24. // Date(const Date& d1)
    25. // {
    26. // _year = d1._year;
    27. // _month = d1._month;
    28. // _day = d1._day;
    29. // }
    30. // Date(Date& d1, Date& d2, Date* d3)
    31. // {
    32. // _year = d1._year;
    33. // _month = d2._month;
    34. // _day = d3->_day;
    35. // }
    36. // Date(const Date* d)
    37. // {
    38. // _year = d->_year;
    39. // _month = d->_month;
    40. // _day = d->_day;
    41. // }
    42. //
    43. //private:
    44. // int _year;
    45. // int _month;
    46. // int _day;
    47. //};
    48. //
    49. //int main()
    50. //{
    51. // A a(1);
    52. // Date c(3, 3, 3);
    53. // Date d(&c);
    54. // Date e(a);
    55. // return 0;
    56. //}
    57. #include<iostream>
    58. using namespace std;
    59. class stack
    60. {
    61. public:
    62. int* _arr;
    63. // 构造函数
    64. stack(int* arr = nullptr)
    65. {
    66. _arr = new int();
    67. int len = sizeof(arr) / sizeof(arr[0]);
    68. copy(arr, arr + len, _arr);
    69. }
    70. ~stack()
    71. {
    72. cout << "删除地址为:" << _arr << "的数组。" << endl;
    73. delete(_arr);
    74. }
    75. 1. 自己开辟一个新空间,然后将内容拷贝到新空间。
    76. //stack(const stack &s)
    77. //{
    78. // //开辟新空间
    79. // _arr = new int();
    80. // //拷贝内容
    81. // int len = sizeof(s._arr) / sizeof(s._arr[0]);
    82. // copy(s._arr, s._arr + len, _arr);
    83. //}
    84. 2. 借助构造函数创建中间对象来实现深拷贝。
    85. //stack(const stack& s)
    86. //{
    87. // //调用构造函数创建中间对象(系统会开辟好空间)
    88. // stack tmp(s._arr);
    89. // swap(_arr, tmp._arr);
    90. //}
    91. 但这有个小问题,就是当前对象未初始化,
    92. 直接交换数值可能会导致程序崩溃,所以加上初始化列表,在交换数值前先初始化。
    93. stack(const stack& s)
    94. :_arr(nullptr)
    95. {
    96. //调用构造函数创建中间对象
    97. stack tmp(s._arr);
    98. swap(_arr, tmp._arr);
    99. }
    100. stack& operator= (const stack& s)
    101. {
    102. // 自己给自己赋值时无需开辟新空间。
    103. if (this != &s)
    104. {
    105. //调用构造函数创建中间对象(系统会开辟好空间)
    106. stack tmp(s._arr);
    107. swap(_arr, tmp._arr);
    108. }
    109. return *this;
    110. }
    111. };
    112. int main()
    113. {
    114. int* arr = new int[5];
    115. arr[0] = 0;
    116. stack st1(arr);
    117. stack st2 = st1;
    118. cout << "st1数组的地址:" << st1._arr << endl \
    119. << "st2数组的地址:" << st2._arr << endl;
    120. return 0;
    121. }

    ------------------------END-------------------------

    才疏学浅,谬误难免,欢迎各位批评指正。

  • 相关阅读:
    每日一题——在windows x86/64 VS环境下,下面的程序会出现什么问题?运行结果是什么?为什么?
    生成元 rust解法
    获取今天包括未来几天数据
    [Linux]线程同步
    Swin Transformer目标检测实验——测试自己的数据集
    【LeetCode刷题-链表】--25.K个一组翻转链表
    Docker 中的端口
    NLP | 注意力机制Attention Mechannism图文详解及代码
    【论文笔记】SDCL: Self-Distillation Contrastive Learning for Chinese Spell Checking
    Python:for循环语句
  • 原文地址:https://blog.csdn.net/look_outs/article/details/129160293