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、40
Person():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