C++零碎记录(四)

6. 深拷贝与浅拷贝

① 浅拷贝:简单的赋值拷贝操作。

② 深拷贝:在堆区重新申请空间,进行拷贝操作。

③ 浅拷贝,如下图所示,带来的问题就是堆区的内存重复释放。

C++零碎记录(四)_第1张图片

④ 深拷贝,如下图所示,在堆区自己创建一份内存,可以避免堆区的内存重复释放。

C++零碎记录(四)_第2张图片

#include 
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 
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 
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 
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

你可能感兴趣的:(C++学习记录,c++,算法,开发语言)