- 什么时候需要拷贝构造函数:
class Test; Test a1;
Test a2 = a1;
Test a3(a1);
void func(Test obj); func(a1);
Test func2(){Test a4; return a4;} - 写一个拷贝构造函数:
要求:第一个参数是 X&, const X&, volatile X&, const volatile X& 之一
没有其他参数,或者其他参数都是默认值
例子:
X::X(const X&){}
X::X(X&, int a=9){}
注意:
一个类可以有多个拷贝构造函数;如果采用X&而不是const X&是不能接受const对像进行调用拷贝构造函数的 - 浅拷贝和深拷贝
用户没写拷贝构造函数,但是发生了1中的操作,会使用编译器提供的默认构造函数(数据成员一一赋值)
浅拷贝局限:static 成员变量, 指针成员变量(一般会动态申请内存)
栗子:
class Test{
public:
static int getCount(){return count;}
private:
static int count; //Test ob2(ob1); Test::getCount()还是等于1,正确应该是2
};
class Test{ //加个拷贝构造函数
Test(const Test& other){count+=1;}
};
class Test{
public:
Test(const Test& other){ //涉及动态申请内存的拷贝构造函数
p = new int();
*p = *(other.p); //类内引用或指针可以直接访问private成员变量
}
private:
int *p;
};
4.技巧
防止对象赋值时候的“值传递”—声明一个私有的空的拷贝构造函数
利用了封装特性,如果用等号赋值,则会调用默认拷贝(值传递), 但是因为默认拷贝认为定义后就不用编译器的了,而且又被定义成了private,所以就没法赋值成功,得到一个编译错误。
class Test{
private:
Test(const Test& other); //没有实现
};
5.拷贝构造函数知识点
- ds