在上一篇博客中,我们学习了构造函数的作用以及它的用法规则。现在,我再为大家介绍一种特殊的构造函数。
我们先来看一段代码:
#include <iostream>
using namespace std;
class Student
{
public:
Student() { cout << "Student()" << endl; }
private:
string m_strName;
};
int main(void)
{
Student stu1;
Student stu2 = stu1;
Student stu3(stu1);
system("PAUSE");
return EXIT_SUCCESS;
}
我们在类中定义了一个构造函数,并且在调用的时候打印一行"Student()"。在main函数中,我们实例化了三个对象,分别为stu1,stu2,stu3。在实例化stu2和stu3这两个对象时,我们对它们用了复制初始化和直接初始化两种初始化方法。按之前讲的知识来说,运行结果应该会打印出三行"Student()"。但是真实的运行结果:
我们看到只出现了一行"Student()",也就是说,构造函数只被调用了一次。那么可能有人会问,这不是与之前讲的“对象实例化的时候一定会调用构造函数”这个说法相违背了吗?
其实这并没有违背,它们其实也调用了构造函数。只不过它调用的不是普通的构造函数,而是“拷贝构造函数”。它是在采用直接初始化或复制初始化实例化对象时,被系统自动调用的一种构造函数。
定义拷贝构造函数的方法和普通构造函数有所不同。它的定义格式时:类名(const 类名& 变量名)。就像这样:
class Student
{
public:
Student() { cout << "Student()" << endl; }
Student(const Student& stu) { cout << "Student(const Student& stu)" << endl; }
private:
string m_strName;
};
这样一来,我们就可以看到运行结果打印出了三行文字:
其中通过后面两行的文字,我们可以知道调用了拷贝构造函数。
和普通构造函数一样,拷贝构造函数没有返回值;如果没有定义拷贝构造函数,系统会自动生成一个不起任何作用的拷贝构造函数;而且在拷贝构造函数后面也是可以跟上初始化列表的。和普通构造函数不同的是,由于拷贝构造函数的参数是唯一确定的,拷贝构造函数不能进行重载。
拷贝构造函数除了在用直接初始化或复制初始化实例化对象的时候会被调用,在参数传递的时候也会被调用。具体是什么意思呢?我们再来举一个例子:
#include <iostream>
using namespace std;
class Student
{
public:
Student() { cout << "Student()" << endl; }
Student(const Student& stu) { cout << "Student(const Student& stu)" << endl; }
private:
string m_strName;
};
void test(Student s) {}
int main(void)
{
Student stu;
test(stu);
system("PAUSE");
return EXIT_SUCCESS;
}
可以看到,我们在中间加入了一个叫做test的不进行任何操作的函数,里面我们定义了一个Student类的参数。接下来,在main函数中实例化一个对象stu,再将这个对象传入刚才定义的函数中,我们来看一下运行结果:
可以看到,在调用了test函数之后,系统也自动调用了拷贝构造函数。这就是在参数传递时调用拷贝构造函数的一个例子。
以上就是有关拷贝构造函数的内容,在下一篇博客中我将介绍析构函数。