一、什么情况下会调用复制构造函数?

1.用同一类型的对象来初始化另一对象

例1:

class A  
{  
public:  
    A(){}  
    A(const A& a){cout<<"copy"<<endl;}  
};  
int main()  
{  
    A a, c;  
    A b = a;//显式复制构造函数(1)  
    c = a;//赋值函数(2)  
    return 0;  
}

输出:

copy

本例要注意的是(1)和(2)的区别:

复制构造函数也是构造函数的一种。只要是构造函数,就要开辟空间。

(1)在初始化的同时还要完成开辟空间的任务,所以在复制构造函数

(2)在L9已经开辟了空间(变通构造函数),这里只是赋值

2.按值传参数或按值传返回值

例2:

class A  
{  
public:  
    A(){}  
    A(const A& a){cout<<"copy"<<endl;}  
};  
A Test(A a)  
{  
    return a;  
}  
int main()  
{  
    A a;  
    Test(a);  
    return 0;  
}

输出:

copy
copy

Test(a)调用了两次构造函数,一次传参,一次是返回值

3.初始化顺序容器中的元素,顺序容包括vector、list、deque

例3:

class A  
{  
    int x;  
public:  
    A(int i):x(i){cout<<"construct "<<i<<endl;}  
    A(const A& a){cout<<"copy"<<endl;}  
};  
int main()  
{  
    list<A> a(10, 5);  
    return 0;  
}

输出:

construct 5
copy
copy
copy
copy
copy
copy
copy
copy
copy
copy

先构造一个临时对象,再把它依次复制到容器中

4.根据元素初始化列表初始化数组元素

例:

class A  
{  
    int x;  
public:  
    A(){cout<<"construct "<<endl;}  
    A(int i):x(i){cout<<"construct "<<i<<endl;}  
    A(const A& a){cout<<"copy"<<endl;}  
};  
int main()  
{  
    cout<<"****Test1****"<<endl;  
    A s1[5];  
    cout<<"****Test2****"<<endl;  
    A s2[5] = {1, 2, 3, 4, 5};  
    cout<<"****Test3****"<<endl;  
    A a(1), b(2), c(3), d(4), e(5);  
    A s[5] = {a, b, c, d, e};  
    return 0;  
}

输出结果:

****Test1****
construct
construct
construct
construct
construct
****Test2****
construct 1
construct 2
construct 3
construct 4
construct 5
****Test3****
construct 1
construct 2
construct 3
construct 4
construct 5
copy
copy
copy
copy
copy

分析结果:

Test1:构造

Test2:构造(隐式类型转换)

Test3:L16是构造L17是复制

默认的复制构造函数

是否由系统提供

如果定义了自己的构造函数,系统还是会提供一个默认的复制构造函数

如果定义了自己的复制构造函数,系统不会提供一个默认的复制构造函数

默认构造函数做什么?

复制构造函数的行为:依次复制每一个非static成员,只是复制内容,称为浅层复制。

如果类中有指针成员,浅层复制会出错。需要定义自己的复制构造函数,使用深层复制。

复制构造函数的特殊用法

如果要禁止复制,可以复制构造函数声明为私有

如果连友元和成员也要禁止复制,可声明为私有,且只声明不定义。

例如iostream类,iostream类禁止复制,它的复制构造函数是私有的

results matching ""

    No results matching ""