目录
4.2 对象的初始化和清理
- 生活中我们买的电子产品都基本会有出厂设置,在某一天我们不用时候也会删除一些自己信息数据保证安全。
- C++中的面向对象来源于生活,每个对象也都会有初始设置以及 对象销毁前的清理数据的设置。
4.2.1 构造函数和析构函数
对象的初始化和清理也是两个非常重要的安全问题。
一个对象或者变量没有初始状态,对其使用后果是未知的。
同样的使用完一个对象和变量,没有及时清理,也会造成一定的安全问题。
C++利用了构造函数和析构函数解决上述问题,这两个函数将会被编译器自动调用,完成对象初始化和清理工作。
对象的初始化和清理工作是编译器强制要我们做的事情,因此如果我们不提供构造和析构,编译器会提供。但编译器提供的构造函数和析构函数是空实现。
- 构造函数:主要作用在于创建对象时为对象的成员属性赋值,构造函数由编译器自动调用,无须手动调用。
- 析构函数:主要作用在于对象销毁前系统自动调用,执行一些清理工作。
构造函数语法:
类名 () { }
- 1. 构造函数,没有返回值也不写void。
- 2. 函数名称与类名相同。
- 3. 构造函数可以有参数,因此可以发生重载。
- 4. 程序在调用对象时候会自动调用构造,无须手动调用,而且只会调用一次。
析构函数语法:
~类名 () { }
- 1. 析构函数,没有返回值也不写void。
- 2. 函数名称与类名相同,在名称前加上符号 ~。
- 3. 析构函数不可以有参数,因此不可以发生重载。
- 4. 程序在对象销毁前会自动调用析构,无须手动调用,而且只会调用一次。
class Person
{
public:
// 构造函数
Person()
{
cout << "Person 构造函数实现" << endl;
}
// 析构函数
~Person()
{
cout << "Person 析构函数实现" << endl;
}
};
void test01 ()
{
Person p;
}
int main() {
test01();
//Person p;
system("pause");
return 0;
}
----------------------------------------------------------------------
Person 构造函数实现
Person 析构函数实现
请按任意键继续. . .
----------------------------------------------------------------------
int main() {
//test01();
Person p;
system("pause");
return 0;
}
----------------------------------------------------------------------
Person 构造函数实现
请按任意键继续. . .
总结:在函数中调用类,类存放于栈区,调用完函数后会释放。
4.2.2 构造函数的分类及调用
构造函数的两种分类方式:
- 按参数分为:有参构造 和 无参构造(默认构造)
- 按类型分为:普通构造 和 拷贝构造
构造函数的三种调用方式:
- 括号法
- 显示法
- 隐式转换法
构造类型:
- 有参构造:参数列表中有参数,称为有参构造
- 无参构造(默认构造):参数列表中没有参数,称为无参构造
- 普通构造:除了拷贝构造以外的构造函数,都叫普通构造
- 拷贝构造:使用常量引用的方式传递,防止对原对象的值的更改。 Person(const Person& p) { }
示例:
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;
};
调用方法:1. 括号法:
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; // 默认构造函数调用
Person p2(10); // 有参构造函数
Person p3(p2); // 拷贝构造函数
cout << "p2年龄: " << p2.age << endl;
cout << "p3年龄: " << p3.age << endl;
// 注意事项1
// 调用默认构造函数时候,不要加().
// 例: 正确:Person p1; 错误:Person p1();
// 因为下面这行代码,编译器会认为是一个函数的声明,不会认为在创建对象
//Person p1();
}
int main() {
test01();
system("pause");
return 0;
}
-------------------------------------------------------------------------
Person 无参构造函数调用
Person 有参构造函数调用
Person 拷贝构造函数调用
p2年龄: 10
p3年龄: 10
Person 析构函数调用
Person 析构函数调用
Person 析构函数调用
请按任意键继续. . .
调用方法:2. 显示法:
匿名对象。特点:1. 当前行执行结束后,系统会立即回收掉匿名对象。
2. 不要利用拷贝构造函数,初始化匿名对象。编译器会认为 Person (p3); 等价于 Person p3;
// 调用
void test01 ()
{
// 2.显示法
Person p2 = Person(10); // 有参构造调用
Person p3 = Person(p2); // 拷贝构造调用
Person(10); // 匿名对象。特点:当前行执行结束后,系统会立即回收掉匿名对象。
cout << "匿名对象会被系统立即回收: " << endl;
}
---------------------------------------------------------------------------
Person 有参构造函数调用
Person 拷贝构造函数调用
Person 有参构造函数调用
Person 析构函数调用
匿名对象会被系统立即回收:
Person 析构函数调用
Person 析构函数调用
请按任意键继续. . .
调用方法:3. 隐式转换法
// 调用
void test01 ()
{
// 1.括号法
/*Person p1; // 默认构造函数调用
Person p2(10); // 有参构造函数
Person p3(p2); // 拷贝构造函数
cout << "p2年龄: " << p2.age << endl;
cout << "p3年龄: " << p3.age << endl;*/
// 注意事项1
// 调用默认构造函数时候,不要加().
// 例: 正确:Person p1; 错误:Person p1();
// 因为下面这行代码,编译器会认为是一个函数的声明,不会认为在创建对象
//Person p1();
// 2.显示法
/*Person p2 = Person(10); // 有参构造调用
Person p3 = Person(p2); // 拷贝构造调用
Person(10); // 匿名对象。特点:当前行执行结束后,系统会立即回收掉匿名对象。
cout << "匿名对象会被系统立即回收: " << endl;*/
// 3.隐式转换法
Person p1 = 10; // 相当于写了 Person p1 = Person(10); 编译器进行的隐式转换
Person p2 = p1;
}
-------------------------------------------------------------------------------
Person 有参构造函数调用
Person 拷贝构造函数调用
Person 析构函数调用
Person 析构函数调用
请按任意键继续. . .
相关教程
- 开发环境搭建:Visual Studio 2019 C++开发环境搭建
- 推荐视频:https://www.bilibili.com/video/BV1et411b73Z?from=search&seid=4205594350351753444
- 已投币三连,非常细致的视频教程,感谢up主。