Copy Constructor——复制构造函数
基本概念
对于复制构造函数,其参数固定只有一个,是对同类对象的引用。
其形式上,为X::X( X & )或X::X(const X &),X::X( X )形式的函数是不存在的。
类似于构造函数,当我们没有定义复制构造函数是,编译器会生成默认的复制构造函数,仅完成完全的复制功能。
调用默认复制构造函数的实例:
class Complex{
private:
double real, imag;
};
Complex c1; //调用默认的构造函数
Complex c2(c1); //调用默认的复制构造函数
调用复制构造函数:
class Complex{
public:
double real, imag;
Complex(){}//构造函数
Complex(const Complex & c){
real = c.real;
imag = c.imag;
std::cout << "Copy Constructor called"<< std::endl;
}
};
Complex c1;
Complex c2(c1);//输出 Copy Constructor called
复制构造函数起作用的三种情况
(1)在使用一个对象来初始化同类的另一个对象时
//两种形式
Complex c2(c1);
Complex c2 = c1;//在定义新对象时,为复制构造的初始化语句,而非赋值语句
(2)当函数中有一个参数为类A 的对象,当该函数被调用时,类A 的复制构造函数将被调用。
class A{
public:
A(){}//构造函数
A(A & a){
std::cout << "Copy Constructor called"<< std::endl;
}
};
void Func(A a1) { }
int main(){
A a2;
Func(a2);//输出 Copy constructor called
return 0;
}
在Func(a2)语句被调用时,传入实参a2,其实是复制出一个对象(形参),放入该函数中运行。
(3)若函数的返回值是类A的对象时,当函数返回时,A的复制构造函数被调用。
class A
{
public:
int v;
A(int n){ v = n;};
A(const A & a){
v = a.v;
std::cout << "Copy constructor called" << std:endl;
}
};
A Func(){
A b(4);
return b;
}
int main(){
std::cout<<Func.v()<<std::endl;
return 0;
}
在Func()运行并产生返回值b时,输出“Copy constructor called”,完成标准流输出语句时,输出“4”
Note:在非初始化(定义)时,对于c2 = c1语句,并不导致复制构造函数的调用,而是简单的赋值,如下例所示。
class CMyclass{
public:
int n;
CMyclass(){}
CMyclass(CMyclass & c){ n =2 * c.n; }
};
int main(){
CMyclass c1,c2;
c1.n = 5;
c2 = c1; //赋值,而非调用复制构造函数
CMyclass c3(c1);
cout <<"c2.n = "<< c2.n << ",";
cout <<"c3.n = "<< c3.n <<endl;
return 0;
}
//输出结果:c2.n = 5,c3.n = 10
小tip:常量引用参数
对于将对象作为实参 CMyclass obj_ 传入函数进行使用的情形,由于复制构造函数调用时的开销比较大 ,故而可以使用其引用 CMyclass & obj_ 作为参数,如果在此基础上,不希望实参受到改变,则在参数前加上const关键字,如下:
void Func( const CMyclass & obj_) { }