当前位置:编程学习 > C/C++ >>

C++拷贝构造函数详解

. 什么是拷贝构造函数
首先对于普通类型的对象来说,它们之间的复制是很简单的,例如:
[c-sharp] 
int a = 100; 
int b = a;  

而类对象与普通对象不同,类对象内部结构一般较为复杂,存在各种成员变量。
下面看一个类对象拷贝的简单例子。
[c-sharp] 
#include <iostream> 
using namespace std; 
 
class CExample { 
private: 
     int a; 
public: 
      //构造函数 
     CExample(int b) 
     { a = b;} 
 
      //一般函数 
     void Show () 
     { 
        cout<<a<<endl; 
      } 
}; 
 
int main() 

     CExample A(100); 
     CExample B = A; //注意这里的对象初始化要调用拷贝构造函数,而非赋值 
      B.Show (); 
     return 0; 

运行程序,屏幕输出100。从以上代码的运行结果可以看出,系统为对象 B 分配了内存并完成了与对象 A 的复制过程。就类对象而言,相同类型的类对象是通过拷贝构造函数来完成整个复制过程的。
下面举例说明拷贝构造函数的工作过程。
[c-sharp] 
#include <iostream> 
using namespace std; 
 
class CExample { 
private: 
    int a; 
public: 
    //构造函数 
    CExample(int b) 
    { a = b;} 
     
    //拷贝构造函数 
    CExample(const CExample& C) 
    { 
        a = C.a; 
    } 
 
    //一般函数 
    void Show () 
    { 
        cout<<a<<endl; 
    } 
}; 
 
int main() 

    CExample A(100); 
    CExample B = A; // CExample B(A); 也是一样的 
     B.Show (); 
    return 0; 
}  

CExample(const CExample& C) 就是我们自定义的拷贝构造函数。可见,拷贝构造函数是一种特殊的构造函数,函数的名称必须和类名称一致,它必须的一个参数是本类型的一个引用变量。

二. 拷贝构造函数的调用时机
在C++中,下面三种对象需要调用拷贝构造函数!
1. 对象以值传递的方式传入函数参数
[c-sharp] 
class CExample  

private: 
 int a; 
 
public: 
 //构造函数 
 CExample(int b) 
 {  
  a = b; 
  cout<<"creat: "<<a<<endl; 
 } 
 
 //拷贝构造 
 CExample(const CExample& C) 
 { 
  a = C.a; 
  cout<<"copy"<<endl; 
 } 
  
 //析构函数 
 ~CExample() 
 { 
  cout<< "delete: "<<a<<endl; 
 } 
 
     void Show () 
 { 
         cout<<a<<endl; 
     } 
}; 
 
//全局函数,传入的是对象 
void g_Fun(CExample C) 

 cout<<"test"<<endl; 

 
int main() 

 CExample test(1); 
 //传入对象 
 g_Fun(test); 
 
 return 0; 

调用g_Fun()时,会产生以下几个重要步骤:
(1).test对象传入形参时,会先会产生一个临时变量,就叫 C 吧。
(2).然后调用拷贝构造函数把test的值给C。 整个这两个步骤有点像:CExample C(test);
(3).等g_Fun()执行完后, 析构掉 C 对象。
2. 对象以值传递的方式从函数返回
[c-sharp] 
class CExample  

private: 
 int a; 
 
public: 
 //构造函数 
 CExample(int b) 
 {  
  a = b; 
 } 
 
 //拷贝构造 
 CExample(const CExample& C) 
 { 
  a = C.a; 
  cout<<"copy"<<endl; 
 } 
 
     void Show () 
     { 
         cout<<a<<endl; 
     } 
}; 
 
//全局函数 
CExample g_Fun() 

 CExample temp(0); 
 return temp; 

 
int main() 

 g_Fun(); 
 return 0; 

当g_Fun()函数执行到return时,会产生以下几个重要步骤:
(1). 先会产生一个临时变量,就叫XXXX吧。
(2). 然后调用拷贝构造函数把temp的值给XXXX。整个这两个步骤有点像:CExample XXXX(temp);
(3). 在函数执行到最后先析构temp局部变量。
(4). 等g_Fun()执行完后再析构掉XXXX对象。
3. 对象需要通过另外一个对象进行初始化;
[c-sharp] 
CExample A(100); 
CExample B = A;  
// CExample B(A);  
后两句都会调用拷贝构造函数。

三. 浅拷贝和深拷贝
1. 默认拷贝构造函数
    很多时候在我们都不知道拷贝构造函数的情况下,传递对象给函数参数或者函数返回对象都能很好的进行,这是因为编译器会给我们自动产生一个拷贝构造函数,这就是“默认拷贝构造函数”,这个构造函数很简单,仅仅使用“老对象”的数据成员的值对“新对象”的数据成员一一进行赋值,它一般具有以下形式:
[c-sharp] 
Rect::Rect(const Rect& r) 

    width = r.width; 
    height = r.height; 

 
    当然,以上代码不用我们编写,编译器会为我们自动生成。但是如果认为这样就可以解决对象的复制问题,那就错了,让我们来考虑以下一段代码:
[c-sharp]
class Rect 

public: 
    Rect()      // 构造函数,计数器加1 
    { 
        count++; 
    } 
    ~Rect()     // 析构函数,计数器减1 
    { 
        count--; 
    } 
    static int getCount()       // 返回计数器的值 
    { 
        return count; 
    } 
private: 
    int width; 
    int height; 
    static int count;       // 一静态成员做为计数器 
}; 
 
int Rect::count = 0;        // 初始化计数器 
 
int main() 

 

补充:软件开发 , C++ ,
CopyRight © 2022 站长资源库 编程知识问答 zzzyk.com All Rights Reserved
部分文章来自网络,