6. 深拷贝与浅拷贝
① 浅拷贝:简单的赋值拷贝操作。
② 深拷贝:在堆区重新申请空间,进行拷贝操作。
③ 浅拷贝,如下图所示,带来的问题就是堆区的内存重复释放。
④ 深拷贝,如下图所示,在堆区自己创建一份内存,可以避免堆区的内存重复释放。
#include <iostream>
using namespace std;class Person
{
public:Person(){cout << "Person的默认构造函数调用" << endl;}Person(int age,int height){m_Age = age;m_Height = new int(height); //把数据创建在堆区,用指针接收new创建的地址cout << "Person的有参构造函数调用" << endl;}//自己实现拷贝函数 解决浅拷贝带来的问题Person(const Person& p){cout << "Person 拷贝构造函数调用" << endl;m_Age = p.m_Age;//m_Height = p.m_Height; 编译器默认实现就是这行代码,默认执行的是浅拷贝//浅拷贝带来的问题就是堆区的内存重复释放// 深拷贝操作,在堆区自己创建一份内存m_Height = new int(*p.m_Height);}~Person(){//析构代码,将堆区开辟数据做释放操作cout << "Person的析构函数调用" << endl;if (m_Height != NULL){delete m_Height; //释放堆区数据}}int m_Age;int * m_Height;
};void test01()
{Person p1(18,160);cout << "p1的年龄为:" << p1.m_Age << "身高为:" << * p1.m_Height << endl; //指针通过解引用获得数据Person p2(p1);cout << "p2的年龄为:" << p2.m_Age << "身高为:" << * p2.m_Height << endl;}int main()
{test01();system("pause");return 0;}
运行结果:
- Person的有参构造函数调用
- p1的年龄为:18身高为:160
- Person 拷贝构造函数调用
- p2的年龄为:18身高为:160
- Person的析构函数调用
- Person的析构函数调用
- 请按任意键继续. . .
7. 初始化列表
① C++提供了初始化列表语法,用来初始化属性。
② 语法:构造函数(): 属性1(值1),属性2(值2),...,()
7.1 传统初始化操作
#include <iostream>
using namespace std;class Person
{
public://传统初始化操作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;
};void test01()
{Person p(10, 20, 30);cout << "m_A:" << p.m_A << endl;cout << "m_B:" << p.m_B << endl;cout << "m_C:" << p.m_C << endl;}int main()
{test01();system("pause");return 0;}
运行结果:
- m_A:10
- m_B:20
- m_C:30
- 请按任意键继续. . .
7.2 灵活初始化操作
#include <iostream>
using namespace std;class Person
{
public:/*构造函数型的初始化操作固定初始化10、30、40Person():m_A(10),m_B(30),m_C(40){}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;
};void test01()
{Person p(30, 20, 10);cout << "m_A:" << p.m_A << endl;cout << "m_B:" << p.m_B << endl;cout << "m_C:" << p.m_C << endl;}int main()
{test01();system("pause");return 0;}
运行结果:
- m_A:30
- m_B:20
- m_C:10
- 请按任意键继续. . .
8. 类对象作为类成员
① C++类中的属性、方法称为成员。
② C++类中的成员可以是另一个类的对象,称该成员为对象成员。
③ B类中有对象A作为成员,A为对象成员,那么当创建B对象时,A与B的构造和析构的顺序是:
1. 当其他类对象作为本类成员,构造时候先构造其他类对象,在构造自身。
2. 当其他类对象作为本类成员,析构的顺序与构造相反,想析构自身,再析构其他类对象。
#include <iostream>
using namespace std;//手机类
class Phone
{
public:Phone(string pName){cout << "Phone的构造函数调用" << endl;m_PName = pName;}~Phone(){cout << "Phone的析构代码函数调用" << endl;}string m_PName;
};//人类
class Person
{
public://m_Phone(pName) 中m_Phone为phone对象,此语句类似于隐式转换法 Phone m_Phone = pName Person(string name, string pName) :m_Name(name), m_Phone(pName) //掉用的是灵活初始化列表 {cout << "Person的构造函数调用" << endl;}~Person(){cout << "Person的析构代码函数调用" << endl;}//姓名string m_Name;//手机Phone m_Phone;
};//当其他类对象作为本类成员,构造时候先构造其他类对象,在构造自身。
//当其他类对象作为本类成员,析构的顺序与构造相反,想析构自身,再析构其他类对象
void test01()
{Person p("张三", "苹果MAX");cout << p.m_Name << "m_A:" << p.m_Phone.m_PName << endl;
}int main()
{test01();system("pause");return 0;}
运行结果:
- Phone的构造函数调用
- Person的构造函数调用
- 张三m_A:苹果MAX
- Person的析构代码函数调用
- Phone的析构代码函数调用
- 请按任意键继续. . .
链接:https://www.zhihu.com/question/437657370/answer/1692846096