目录
对象的初始化和清理
c++每个对象都会有初始设置以及对象销毁前的清理数据的设置
构造函数和析构函数:
对象的初始化和清理是两个非常重要的安全问题
一个对象或者变量没有初始状态,对其使用后果是未知
使用完一个对象或者变量没有及时清理,也会造成一定的安全问题
c++利用构造函数和析构函数解决上述问题,这两个函数将会被编译器自动调用,完成对象初始化和清理工作
对象的初始化和清理工作是编译器强制我们做的事,如果我们不提供构造和析构,编译器会提供
编译器提供的构造函数和析构函数是空实现
构造函数:创建对象时为对象的成员属性赋值,由编译器自动调用,无需手动调用
析构函数:对象销毁前系统自动调用,执行一些清理工作
构造函数语法:
类名(){}
1、构造函数没有返回值也不需要写void
2、函数名称与类名相同
3、可以有参数,可以发生重载
4、会自动调用,无需手动,且只会调用一次
析构函数语法:
~类名(){}
1、没有返回值也不需要写void
2、函数名称与类名相同,在名称前加~
3、不可以有参数,不能重载
4、会自动调用,无需手动,且只会调用一次
构造函数的分类及调用
两种分类方式:
按参数分为:有参构造和无参构造(默认构造)
按类型分为:普通构造和拷贝构造
三种调用方式:
括号法
显示法
隐式转换法
示例:
#include<iostream>
using namespace std;
class Person {
public:
// 构造函数
// 无参构造函数
Person() {
cout << "person类构造函数" << endl;
}
// 有参构造函数
Person(int a) {
age = a;
cout << "person类有参构造函数" << endl;
}
//拷贝构造 拷贝一个相同的对象 用引用,传值会创建副本,再次调用拷贝构造函数,形成死循环
Person(const Person& p) {
age = p.age; //将属性传到调用此函数的部分
cout << "person类拷贝构造函数" << endl;
}
// 析构函数
~Person() {
cout << "person类析构函数" << endl;
}
int age;
};
//调用
void test01() {
//1、括号法
Person p1; //栈上的数据,test01执行完毕后释放这个对象
Person p2(10);//有参构造
Person p3(p1);//拷贝构造
//注意事项
//调用默认构造函数的时候不要加() Person p1();编译器会认为这是函数的声明,不会创建对象
//2、显示法
Person p4; //无参构造
Person p5 = Person(10); //有参构造
Person p6 = Person(p5); //拷贝构造
Person(10);//匿名对象 特点:当前行执行结束后系统会立即回收匿名对象
//注意:不要利用拷贝对象初始化匿名对象 eg:Person(p5); 编译器认为Person(p5)等价于Person p5;是对象的声明
//3、隐式转换法
Person p7 = 10; // 相当于写了Person p7 = Person(10); 有参构造
Person p8 = p7; //拷贝构造
}
int main() {
test01();//栈上的数据,test01执行完毕后释放这个对象
Person p2;
system("pause");
return 0;
}
拷贝构造函数调用时机:
1、使用一个已经创建完毕的对象来初始化一个新对象
2、值传递的方式给函数参数传值
3、值方式return返回局部对象
构造函数调用规则
默认情况下,c++编译器至少给一个类添加三个函数
1、默认构造函数(无参,函数体为空)
2、默认析构函数(无参,函数体为空)
3、默认拷贝构造函数,对属性进行值拷贝
调用规则:
如果用户定义有参构造函数,c++不再提供默认无参构造,但是会提供默认拷贝构造
如果用户定义拷贝构造函数,c++不会再提供其他构造函数
深拷贝和浅拷贝
浅拷贝:简单的赋值拷贝操作,指向同一片空间 问题:堆区的内存重复释放,要利用深拷贝进行解决
深拷贝:在堆区重新申请空间,进行拷贝操作,将数据放在重新申请的空间
拷贝构造函数中:
m_height = new int(*p.m_height);
析构中:
If(m_height ! =NULL)
{delete m_height;
m_height=NULL;
}
如果属性有在堆区开辟的,一定要自己提供拷贝构造函数,防止浅拷贝带来的问题
初始化列表
作用:初始化属性
语法:构造函数():属性1(值1),属性2(值2)…{}
Class Person{
Public:
Person():m_a(10),m_b(20),m_c(30)
{
}
Int m_a;
Int m_b;
Int m_c;
};
改进:
Person(int a,int b,int c):m_a(a),m_b(b),m_c(c)
{
}
Int m_a;
Int m_b;
Int m_c;
Person p(30,20,10);