1、构造函数
构造函数函数名和类名相同,没有返回类型,连void都不可以,但可以有参数,可以重载
构造函数:实例化对象的时候系统自动调用
2、析构函数
析构函数函数名是在类名前面加”~”组成,没有返回类型,连void都不可以,不能有参数,不能重载
析构函数:对象释放的时候系统自动调用
案例:
class Data
{
private:
int num;
public:
//构造函数(无参的构造)
Data()
{
num = 0;
cout<<"无参的构造函数"<<endl;
}
//构造函数(有参的构造)
Data(int n)
{
num = n;
cout<<"有参的构造函数"<<endl;
}
//析构函数
~Data()
{
cout<<"析构函数"<<endl;
}
};
void test01()
{
//类实例化对象 系统自定调用构造函数
Data ob;
//函数结束的时候 局部对象ob 被释放 系统自动调用析构函数
}
int main(int argc, char *argv[])
{
cout<<"-----001-----"<<endl;
test01();
cout<<"------002-------"<<endl;
return 0;
}
3、构造函数分类:
按参数类型:分为无参构造函数 和有参构造函数
按类型分类:分为普通构造函数 和拷贝构造函数 (复制构造函数)
4、构造函数的调用
//调用无参 或 默认构造 (隐式调用)
Data ob1;
//调用无参构造 (显示调用)
Data ob2 = Data();
//调用有参构造(隐式调用)
Data ob3(10);
//调用有参构造(显示调用)
Data ob4 = Data(20);
//隐式转换的方式 调用有参构造(针对于 只有一个数据成员)(尽量别用)
Data ob5 = 30;//转化成Data ob5(30)
//匿名对象(当前语句结束 匿名对象立即释放)
Data(40);
注意:在同一作用域 构造和析构的顺序相反
5、拷贝构造函数
也就是,旧对象 初始化 新对象 才会调用拷贝构造函数
(系统提供一个拷贝构造函数 赋值操作)
//拷贝构造函数
Data(const Data &ob)//const Data &ob = ob1
{
//拷贝构造函数 是ob2调用 num就是ob2的num
//ob2.num = ob1.num
num = ob.num;
cout<<"拷贝构造"<<endl;
}
拷贝构造函数的调用
//调用拷贝构造函数(如果用户 不实现拷贝构造 系统将调用默认的拷贝构造)
//默认的拷贝构造:单纯的整体赋值(浅拷贝)
//如果用户实现了 拷贝构造 系统将调用用户实现的拷贝构造
Data ob1(10);
cout<<"ob1.num = "<<ob1.num<<endl;
Data ob2(ob1);//隐式调用拷贝构造函数
cout<<"ob2.num = "<<ob2.num<<endl;
Data ob3 = Data(ob1);//显示调用拷贝构造函数
cout<<"ob3.num = "<<ob3.num<<endl;
Data ob4 = ob1;//=隐式转换调用
cout<<"ob4.num = "<<ob4.num<<endl;
下面的方式不会调用拷贝构造函数
Data ob1(10);
Data ob2;
ob2 = ob1;//不会调用拷贝构造 单纯对象 赋值操作
6、拷贝构造函数的注意事项:
1、不能调用拷贝构造函数去初始化匿名对象,也就是说以下代码不正确
2、对象作为函数的参数 如果实参与形参 都是普通对象 那么就会调用拷贝构造函数
3、函数返回局部对象 在qt中会被优化 从而调用不了拷贝构造函数