• C++ 浅拷贝与深拷贝探究


    C++浅拷贝与深拷贝探究


    浅拷贝与深拷贝的概念是在类的复制/拷贝构造函数中出现的。

    拷贝构造函数使用场景

    • 对象作为参数,以值传递方式传入函数(要调用拷贝构造函数将实参拷贝给函数栈中的形参)
    • 对象作为返回值,以值方式返回(要调用拷贝构造函数,将要传出的对象拷贝给一个外作用域的临时对象)
    • 用一个对象去初始化另外一个对象

    浅拷贝:直接使用 “=” 赋值,拷贝给定对象属性。(如果是一般类型,那么就相当于赋值运算符。如果有指针,赋值后就相当于拷贝双方指向同一个地址

    private:

    xxx *pStr;

    ......

    pStr = m.pStr;//拷贝的是地址

    深拷贝:在赋值之前,先给指针申请一段内存,而后拷贝给定对象属性值。(即:先申请内存,再赋值

    pStr = new xxx[xxx];//先申请内存

    对pStr赋值//再拷贝值

    实例说明

    浅拷贝

     1 #pragma once
     2 class Demo
     3 {
     4 private:
     5     int m_icount;
     6     int *pStr;
     7 public:
     8     Demo(int icount);//构造函数
     9     Demo(const Demo &d);//拷贝构造函数(浅拷贝)
    10     ~Demo();//析构函数
    11     void setValue(int icount);
    12     void printValue();
    13     void printAddr();
    14 };
    Demo.h
     1 #include "Demo.h"
     2 #include <iostream>
     3 using namespace std;
     4 
     5 Demo::Demo(int icount)
     6 {
     7     m_icount = icount;
     8     pStr = new int[m_icount];
     9     for (int i = 0; i < m_icount; i++)
    10     {
    11         pStr[i] = i;
    12     }
    13     cout << "
    Demo(int value)构造函数被调用!" << endl;
    14 }
    15 
    16 Demo::Demo(const Demo &d)
    17 {
    18     m_icount = d.m_icount;
    19     //浅拷贝
    20     pStr = d.pStr;//浅拷贝,赋地址
    21     cout << "Demo(const Demo &d)浅拷贝构造函数被调用!" << endl;
    22 
    23     //深拷贝
    24     /*pStr = new int[m_icount];
    25     for (int i = 0; i < m_icount; i++)
    26     {
    27         pStr[i] = d.pStr[i];
    28     }
    29     cout << "
    Demo(const Demo &d)深拷贝构造函数被调用!" << endl;*/
    30 }
    31 
    32 Demo::~Demo()
    33 {
    34     delete[]pStr;
    35     pStr = NULL;
    36     cout << "
    ~Demo()析构函数被调用!" << endl;
    37 }
    38 
    39 void Demo::setValue(int icount)
    40 {
    41     m_icount = icount;
    42     cout << "
    setValue(int icount)函数被调用!" << endl;
    43 }
    44 
    45 void Demo::printValue()
    46 {
    47     for (int i = 0; i < m_icount; i++)
    48     {
    49         cout << pStr[i] << "  ";
    50     }
    51     cout << "
    Demo::printValue()函数被调用!" << endl;
    52 }
    53 
    54 void Demo::printAddr()
    55 {
    56     cout << pStr << endl;
    57 }
    58 
    59 int main()
    60 {
    61     Demo d1(6);
    62     cout << "
    d1 : ";
    63     d1.printValue();
    64 
    65     d1.setValue(5);
    66     cout << "
    d1 : ";
    67     d1.printValue();
    68 
    69     Demo d2(d1);
    70     cout << "
    d2 :";
    71     d2.printValue();
    72 
    73     cout << "
    d1 addr : ";
    74     d1.printAddr();
    75     cout << "
    d2 addr : ";
    76     d2.printAddr();
    77 
    78     system("pause");
    79 }
    Demo.cpp

     //代码中的拷贝构造函数形参 const Demo &d 

    //const: 禁止在传参过程中改变参数对象

    //&d:引用传递,参数如果不是引用传递,那么在将实参拷贝给形参时就要调用拷贝构造函数,而函数自身就是拷贝构造函数,就会不断调用自己,陷入递归,无法自拔。->拷贝构造函数必须使用引用传递!!!

    结果

    那么,对于两者中任意一个对象进行操作就势必会改变另外一个,这是我们一般不希望看到的。这时候,我们就需要深拷贝。为即将产生的新对象重新申请存储空间。

    深拷贝

     1 #include "Demo.h"
     2 #include <iostream>
     3 using namespace std;
     4 
     5 Demo::Demo(int icount)
     6 {
     7     m_icount = icount;
     8     pStr = new int[m_icount];
     9     for (int i = 0; i < m_icount; i++)
    10     {
    11         pStr[i] = i;
    12     }
    13     cout << "
    Demo(int value)构造函数被调用!" << endl;
    14 }
    15 
    16 Demo::Demo(const Demo &d)
    17 {
    18     m_icount = d.m_icount;
    19     //浅拷贝
    20     /*pStr = d.pStr;//浅拷贝,赋地址
    21     cout << "Demo(const Demo &d)浅拷贝构造函数被调用!" << endl;*/
    22 
    23     //深拷贝
    24     pStr = new int[m_icount];
    25     for (int i = 0; i < m_icount; i++)
    26     {
    27         pStr[i] = d.pStr[i];
    28     }
    29     cout << "
    Demo(const Demo &d)深拷贝构造函数被调用!" << endl;
    30 }
    31 
    32 Demo::~Demo()
    33 {
    34     delete[]pStr;
    35     pStr = NULL;
    36     cout << "
    ~Demo()析构函数被调用!" << endl;
    37 }
    38 
    39 void Demo::setValue(int icount)
    40 {
    41     m_icount = icount;
    42     cout << "
    setValue(int icount)函数被调用!" << endl;
    43 }
    44 
    45 void Demo::printValue()
    46 {
    47     for (int i = 0; i < m_icount; i++)
    48     {
    49         cout << pStr[i] << "  ";
    50     }
    51     cout << "
    Demo::printValue()函数被调用!" << endl;
    52 }
    53 
    54 void Demo::printAddr()
    55 {
    56     cout << pStr << endl;
    57 }
    58 
    59 int main()
    60 {
    61     Demo d1(6);
    62     cout << "
    d1 : ";
    63     d1.printValue();
    64 
    65     d1.setValue(5);
    66     cout << "
    d1 : ";
    67     d1.printValue();
    68 
    69     Demo d2(d1);
    70     cout << "
    d2 :";
    71     d2.printValue();
    72 
    73     cout << "
    d1 addr : ";
    74     d1.printAddr();
    75     cout << "
    d2 addr : ";
    76     d2.printAddr();
    77 
    78     system("pause");
    79 }
    Demo.cpp

    结果

    简言之,浅拷贝就是把别人的地址拿过来作为自己的,双方共同维护一个对象,“绑在一块,一损俱损”;而深拷贝则是赋值之前为新对象申请了额外的空间,所以独立于被拷贝对象,“咱没关系,爱咋咋地”!

    拷贝构造函数使用原则

    • 含有指针类型的成员或者含有动态内存空间的对象,必须提供自定义的拷贝构造函数。
    • 在提供拷贝构造函数的同时,也要考虑实现自定义的赋值运算符。

    怎样区别

    赋值运算还是拷贝函数?有无新对象产生

    深拷贝还是浅拷贝?有无指针成员和动态内存空间

  • 相关阅读:
    Java作业十(2017-11-8)
    Java作业九(2017-11-6)
    Java作业八(2017-10-30)
    Java作业七(2017-10-30)
    Java作业六(2017-10-30)
    Java作业五(2017-10-15)
    如何获取jqGrid中选择的行的数据
    如何修改WAMP中mysql默认空密码
    为 PhpStorm 配置 Xdebug 来调试代码
    MySQL load_file()/into outfile路径问题总结
  • 原文地址:https://www.cnblogs.com/yocichen/p/10359542.html
Copyright © 2020-2023  润新知