本文讲解了C++中构造函数的种类
一、无参构造函数
- 如果没有定义构造函数,则系统自动调用此默认构造函数,且什么都不做。
- 如果用户自定义了带参数的构造函数,若还想调用无参的构造函数,必须显示定义
person() {
cout << "this object is being created." << endl;
}
二、含参构造函数
一般构造函数可以有各种参数形式,一个类可以有多个一般构造函数,前提是参数的个数或者类型不同(基于c++的重载函数原理)
person(string ID, int age, int height, string sex) {
cout<<"含参构造函数调用"<< endl;
this.ID = ID;
this.age = age;
this.height = height;
this.sex = sex;
}
含参的构造函数还可以使用快速初始化的方法:
person(string ID, int age, int height, string sex) : ID(ID), age(age), height(height), sex(sex) { }
注意:
- 初始化参数列表和初始化顺序不可以变动
- 变量名称(不是参数列表名称)一定要和成员变量名一样
三、拷贝构造函数
使用已经创建好的同一类对象,进行构造另一个对象。
person(const person &p) {
cout << "拷贝构造函数调用" << endl;
this->ID = p.ID;
this->age = p.age;
this->height = p.height;
this->sex = p.sex;
}
注意:
- 但当类中有指针成员时,由系统默认创建该复制构造函数会存在风险,具体原因请查询有关 “浅拷贝“
四、深拷贝和浅拷贝
浅拷贝,指的是在对象复制时,只对对象中的数据成员进行简单的赋值,默认拷贝构造函数执行的也是浅拷贝。
- 例如: person p2 = p1; 这就是浅拷贝,将p1中的成员变量值赋给p2,在这里其实就相当于拷贝构造函数的调用。
但是,浅拷贝,在成员变量是引用类型(指针)时,会出现问题。
- 传参问题中,地址传递后可以实现swap函数的调用,这就是浅拷贝的原理
- 当我们使用浅拷贝时,引用类型传递的依旧是地址,这也就是说,拷贝后的对象和拷贝的对象中引用对象指向同一块地址,这样就出现问题了。
在“深拷贝”的情况下,对于对象中动态成员,就不能仅仅简单地赋值了,而应该重新动态分配空间。
如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝。
上面提到,如果没有自定义复制构造函数,则系统会创建默认的复制构造函数,但系统创建的默认复制构造函数只会执行“浅拷贝”,即将被拷贝对象的数据成员的值一一赋值给新创建的对象,若该类的数据成员中有指针成员,则会使得新的对象的指针所指向的地址与被拷贝对象的指针所指向的地址相同,delete该指针时则会导致两次重复delete而出错。
|