首先,什么是浅拷贝?什么是深拷贝?
浅拷贝:在C++中,在用一个对象时,只是复制了成员,并没有复制资源,使两个对象同时指向了同一资源的复制方式称为浅拷贝;(简单的来说就是成员数据间的赋值—数据拷贝)
深拷贝:是指向内容复制到当前对象新分配的缓存缓冲区中的一种复制方式。(—地址拷贝)
我们在学习编程语言过程中常见的拷贝方式如下这种,这是简单的普通对象拷贝:
int a=12;
int b=a;
而深拷贝和 浅拷贝是类对象之间的拷贝,入如下是关于类对象简单的程序:

#include<iostream>
using namespace std;
class Data
{
private:
    int a;
public:
    Data(int b)
    {
        a = b; }
    void show()
    {
        cout << a << endl;

    }
};
int main()
{
    Data A(1112);
    Data B = A;
    B.show();
    return 0;
}

则输出结果为1112;读代码的过程中我们可以看到,系统为对象B分配了内存,并且与对象A进行拷贝;
下面是拷贝构造函数的过程

#include <iostream>
 using namespace std; 
 class Data{
 private:
     int a;
 public:
     Data(int b)
     { a=b;}

    Data(const Data& C)
     {
         a=C.a;
     }
     void Show ()
     {
         cout<<a<<endl;
     }
 }; 
 int main()
 {
     Data A(100);
     Data B=A;
     B.Show ();
     return 0;
 }

Data(const Data& C)就是我们自定义的拷贝构造函数。可见,拷贝构造函数是一种特殊的构造函数,函数的名称必须和类名称一致,它的唯一的一个参数是本类型的一个引用变量,该参数是const类型,不可变的。
当用一个已初始化过了的自定义类类型对象去初始化另一个新构造的对象的时候,拷贝构造函数就会被自动调用。也就是说,当类的对象需要拷贝时,拷贝构造函数将会被调用。以下情况都会调用拷贝构造函数:
一个对象以值传递的方式传入函数体
一个对象以值传递的方式从函数返回
一个对象需要通过另外一个对象进行初始化。

如果在类中没有显式地声明一个拷贝构造函数,那么,编译器将会自动生成一个默认的拷贝构造函数,该构造函数完成对象之间的位拷贝。位拷贝又称浅拷贝,后面将进行说明。

自定义拷贝构造函数是一种良好的编程风格,它可以阻止编译器形成默认的拷贝构造函数,提高源码效率。

浅拷贝和深拷贝

  在某些状况下,类内成员变量需要动态开辟堆内存,如果实行位拷贝,也就是把对象里的值完全复制给另一个对象,如A=B。这时,如果B中有一个成员变量指针已经申请了内存,那A中的那个成员变量也指向同一块内存。这就出现了问题:当B把内存释放了(如:析构),这时A内的指针就是野指针了,出现运行错误。

  深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,就是浅拷贝。下面举个深拷贝的例子。

#include <iostream>
using namespace std;
class str{
 public:
  str(int b,char* cstr)
  {
   a=b;
   str=new char[b];
   strcpy(str,cstr);
  }
  str(const str& C)
  {
   a=C.a;
   str=new char[a]; //深拷贝
   if(str!=0)
    strcpy(str,C.str);
  }
  void Show()
  {
   cout<<str<<endl;
  }
  ~str()
  {
   delete str;
  }
 private:
  int a;
  char *str;
};

int main()
{
 str A(10,"Hello!");
 str B=A;
 B.Show();
 return 0;
}

深拷贝和浅拷贝的定义可以简单理解成:如果一个类拥有资源(堆,或者是其它系统资源),当这个类的对象发生复制过程的时候,这个过程就可以叫做深拷贝,反之对象存在资源,但复制过程并未复制资源的情况视为浅拷贝。

浅拷贝资源后在释放资源的时候会产生资源归属不清的情况导致程序运行出错。

Test(Test &c_t)是自定义的拷贝构造函数,拷贝构造函数的名称必须与类名称一致,函数的形式参数是本类型的一个引用变量,且必须是引用。

当用一个已经初始化过了的自定义类类型对象去初始化另一个新构造的对象的时候,拷贝构造函数就会被自动调用,如果你没有自定义拷贝构造函数的时候,系统将会提供给一个默认的拷贝构造函数来完成这个过程,上面代码的复制核心语句就是通过Test(Test &c_t)拷贝构造函数内的p1=c_t.p1;语句完成的。