21 C++ - 对象的构造和析构

简介: 21 C++ - 对象的构造和析构

1. 初始化和清理

我们大家在购买一台电脑或者手机,或者其他的产品,这些产品都有一个初始设置,也就是这些产品对被创建的时候会有一个基础属性值。那么随着我们使用手机和电脑的时间越来越久,那么电脑和手机会慢慢被我们手动创建很多文件数据,某一天我们不用手机或电脑了,那么我们应该将电脑或手机中我们增加的数据删除掉,保护自己的信息数据。

从这样的过程中,我们体会一下,所有的事物在起初的时候都应该有个初始状态,当这个事物完成其使命时,应该及时清除外界作用于上面的一些信息数据。

那么我们c++中OO思想也是来源于现实,是对现实事物的抽象模拟,具体来说,当我们创建对象的时候,这个对象应该有一个初始状态,当对象销毁之前应该销毁自己创建的一些数据。

对象的初始化和清理也是两个非常重要的安全问题,一个对象或者变量没有初始时,对其使用后果是未知,同样的使用完一个变量,没有及时清理,也会造成一定的安全问题。c++为了给我们提供这种问题的解决方案,构造函数和析构函数,这两个函数将会被编译器自动调用,完成对象初始化和对象清理工作。

无论你是否喜欢,对象的初始化和清理工作是编译器强制我们要做的事情,即使你不提供初始化操作和清理操作,编译器也会给你增加默认的操作,只是这个默认初始化操作不会做任何事,所以编写类就应该顺便提供初始化函数。

为什么初始化操作是自动调用而不是手动调用?既然是必须操作,那么自动调用会更好,如果靠程序员自觉,那么就会存在遗漏初始化的情况出现。

2. 构造函数和析构函数

构造函数主要作用在于创建对象时为对象的成员属性赋值,构造函数由编译器自动调用,无须手动调用。

析构函数主要用于对象销毁前系统自动调用,执行一些清理工作。

构造函数语法:

构造函数函数名和类名相同,没有返回值,不能有void,但可以有参数。


ClassName(){}

析构函数语法:

析构函数函数名是在类名前面加”~”组成,没有返回值,不能有void,不能有参数,不能重载。


~ClassName(){}

class Person{
public:
  Person(){
    cout << "构造函数调用!" << endl;
    pName = (char*)malloc(sizeof("John"));
    strcpy(pName, "John");
    mTall = 150;
    mMoney = 100;
  }
  ~Person(){
    cout << "析构函数调用!" << endl;
    if (pName != NULL){
      free(pName);
      pName = NULL;
    }
  }
public:
  char* pName;
  int mTall;
  int mMoney;
};
void test(){
  Person person;
  cout << person.pName << person.mTall << person.mMoney << endl;
}

3.构造函数的分类及调用

按参数类型: 分为无参构造函数和有参构造函数

按类型分类: 普通构造函数和拷贝构造函数(复制构造函数)

class Person{
public:
  Person(){
    cout << "no param constructor!" << endl;
    mAge = 0;
  }
  //有参构造函数
  Person(int age){
    cout << "1 param constructor!" << endl;
    mAge = age;
  }
  //拷贝构造函数(复制构造函数) 使用另一个对象初始化本对象
  Person(const Person& person){
    cout << "copy constructor!" << endl;
    mAge = person.mAge;
  }
  //打印年龄
  void PrintPerson(){
    cout << "Age:" << mAge << endl;
  }
private:
  int mAge;
};
//1. 无参构造调用方式
void test01(){
  //调用无参构造函数
  Person person1; 
  person1.PrintPerson();
  //无参构造函数错误调用方式
  //Person person2();
  //person2.PrintPerson();
}
//2. 调用有参构造函数
void test02(){
  //第一种 括号法,最常用
  Person person01(100);
  person01.PrintPerson();
  //调用拷贝构造函数
  Person person02(person01);
  person02.PrintPerson();
  //第二种 匿名对象(显示调用构造函数)
  Person(200); //匿名对象,没有名字的对象
  Person person03 = Person(300);
  person03.PrintPerson();
  //注意: 使用匿名对象初始化判断调用哪一个构造函数,要看匿名对象的参数类型
  Person person06(Person(400)); //等价于 Person person06 = Person(400);
  person06.PrintPerson();
  //第三种 =号法 隐式转换
  Person person04 = 100; //Person person04 =  Person(100)
  person04.PrintPerson();
  //调用拷贝构造
  Person person05 = person04; //Person person05 =  Person(person04)
  person05.PrintPerson();
}

b为A的实例化对象,A a = A(b) 和 A(b)的区别?

A(b) 有变量来接的时候,那么编译器认为他是一个匿名对象,当没有变量来接的时候,编译器认为你A(b) 等价于 A b.

注意:不能调用拷贝构造函数去初始化匿名对象,也就是说以下代码不正确:

class Teacher{
public:
  Teacher(){
    cout << "默认构造函数!" << endl;
  }
  Teacher(const Teacher& teacher){
    cout << "拷贝构造函数!" << endl;
  }
public:
  int mAge;
};
void test(){
  Teacher t1;
  //error C2086:“Teacher t1”: 重定义
  Teacher(t1);  //此时等价于 Teacher t1;
}

4. 拷贝构造函数的调用时机

  • 对象以值传递的方式传给函数参数
  • 函数局部对象以值传递的方式从函数返回(vs debug模式下调用一次拷贝构造,qt不调用任何构造)
  • 用一个对象初始化另一个对象
class Person{
public:
  Person(){
    cout << "no param contructor!" << endl;
    mAge = 10;
  }
  Person(int age){
    cout << "param constructor!" << endl;
    mAge = age;
  }
  Person(const Person& person){
    cout << "copy constructor!" << endl;
    mAge = person.mAge;
  }
  ~Person(){
    cout << "destructor!" << endl;
  }
public:
  int mAge;
};
//1. 旧对象初始化新对象
void test01(){
  Person p(10);
  Person p1(p);
  Person p2 = Person(p);
  Person p3 = p; // 相当于Person p2 = Person(p);
}
//2. 传递的参数是普通对象,函数参数也是普通对象,传递将会调用拷贝构造
void doBussiness(Person p){}
void test02(){
  Person p(10);
  doBussiness(p);
}
//3. 函数返回局部对象
Person MyBusiness(){
  Person p(10);
  cout << "局部p:" << (int*)&p << endl;
  return p;
}
void test03(){
  //vs release、qt下没有调用拷贝构造函数
  //vs debug下调用一次拷贝构造函数
  Person p = MyBusiness();
  cout << "局部p:" << (int*)&p << endl;
}

Test03结果说明:

编译器存在一种对返回值的优化技术,RVO(Return Value Optimization).在vs debug模式下并没有进行这种优化,所以函数MyBusiness中创建p对象,调用了一次构造函数,当编译器发现你要返回这个局部的对象时,编译器通过调用拷贝构造生成一个临时Person对象返回,然后调用p的析构函数。

我们从常理来分析的话,这个匿名对象和这个局部的p对象是相同的两个对象,那么如果能直接返回p对象,就会省去一个拷贝构造和一个析构函数的开销,在程序中一个对象的拷贝也是非常耗时的,如果减少这种拷贝和析构的次数,那么从另一个角度来说,也是编译器对程序执行效率上进行了优化。

所以在这里,编译器偷偷帮我们做了一层优化:

  • 当我们这样去调用: Person p = MyBusiness();
  • 编译器偷偷将我们的代码更改为:
void MyBussiness(Person& _result){
       _result.X:X(); //调用Person默认拷贝构造函数
       //.....对_result进行处理
       return;
   }
int main(){
   Person p; //这里只分配空间,不初始化
   MyBussiness(p);
}

5. 构造函数调用规则

默认情况下,c++编译器至少为我们写的类增加3个函数

  1. 默认构造函数(无参,函数体为空)
  2. 默认析构函数(无参,函数体为空)
  3. 默认拷贝构造函数,对类中非静态成员属性简单值拷贝

如果用户定义拷贝构造函数,c++不会再提供任何默认构造函数

如果用户定义了普通构造(非拷贝),c++不在提供默认无参构造,但是会提供默认拷贝构造

目录
相关文章
|
编译器 C++ 开发者
【C++篇】深度解析类与对象(下)
在上一篇博客中,我们学习了C++的基础类与对象概念,包括类的定义、对象的使用和构造函数的作用。在这一篇,我们将深入探讨C++类的一些重要特性,如构造函数的高级用法、类型转换、static成员、友元、内部类、匿名对象,以及对象拷贝优化等。这些内容可以帮助你更好地理解和应用面向对象编程的核心理念,提升代码的健壮性、灵活性和可维护性。
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
编译器 C++
类和对象(中 )C++
本文详细讲解了C++中的默认成员函数,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载和取地址运算符重载等内容。重点分析了各函数的特点、使用场景及相互关系,如构造函数的主要任务是初始化对象,而非创建空间;析构函数用于清理资源;拷贝构造与赋值运算符的区别在于前者用于创建新对象,后者用于已存在的对象赋值。同时,文章还探讨了运算符重载的规则及其应用场景,并通过实例加深理解。最后强调,若类中存在资源管理,需显式定义拷贝构造和赋值运算符以避免浅拷贝问题。
|
存储 编译器 C++
类和对象(上)(C++)
本篇内容主要讲解了C++中类的相关知识,包括类的定义、实例化及this指针的作用。详细说明了类的定义格式、成员函数默认为inline、访问限定符(public、protected、private)的使用规则,以及class与struct的区别。同时分析了类实例化的概念,对象大小的计算规则和内存对齐原则。最后介绍了this指针的工作机制,解释了成员函数如何通过隐含的this指针区分不同对象的数据。这些知识点帮助我们更好地理解C++中类的封装性和对象的实现原理。
|
编译器 C++
类和对象(下)C++
本内容主要讲解C++中的初始化列表、类型转换、静态成员、友元、内部类、匿名对象及对象拷贝时的编译器优化。初始化列表用于成员变量定义初始化,尤其对引用、const及无默认构造函数的类类型变量至关重要。类型转换中,`explicit`可禁用隐式转换。静态成员属类而非对象,受访问限定符约束。内部类是独立类,可增强封装性。匿名对象生命周期短,常用于临时场景。编译器会优化对象拷贝以提高效率。最后,鼓励大家通过重复练习提升技能!
|
安全 编译器 C语言
【C++篇】深度解析类与对象(中)
在上一篇博客中,我们学习了C++类与对象的基础内容。这一次,我们将深入探讨C++类的关键特性,包括构造函数、析构函数、拷贝构造函数、赋值运算符重载、以及取地址运算符的重载。这些内容是理解面向对象编程的关键,也帮助我们更好地掌握C++内存管理的细节和编码的高级技巧。
|
存储 程序员 C语言
【C++篇】深度解析类与对象(上)
在C++中,类和对象是面向对象编程的基础组成部分。通过类,程序员可以对现实世界的实体进行模拟和抽象。类的基本概念包括成员变量、成员函数、访问控制等。本篇博客将介绍C++类与对象的基础知识,为后续学习打下良好的基础。
|
编译器 C++ 容器
【c++11】c++11新特性(上)(列表初始化、右值引用和移动语义、类的新默认成员函数、lambda表达式)
C++11为C++带来了革命性变化,引入了列表初始化、右值引用、移动语义、类的新默认成员函数和lambda表达式等特性。列表初始化统一了对象初始化方式,initializer_list简化了容器多元素初始化;右值引用和移动语义优化了资源管理,减少拷贝开销;类新增移动构造和移动赋值函数提升性能;lambda表达式提供匿名函数对象,增强代码简洁性和灵活性。这些特性共同推动了现代C++编程的发展,提升了开发效率与程序性能。
498 12
|
11月前
|
人工智能 机器人 编译器
c++模板初阶----函数模板与类模板
class 类模板名private://类内成员声明class Apublic:A(T val):a(val){}private:T a;return 0;运行结果:注意:类模板中的成员函数若是放在类外定义时,需要加模板参数列表。return 0;
265 0
|
11月前
|
存储 编译器 程序员
c++的类(附含explicit关键字,友元,内部类)
本文介绍了C++中类的核心概念与用法,涵盖封装、继承、多态三大特性。重点讲解了类的定义(`class`与`struct`)、访问限定符(`private`、`public`、`protected`)、类的作用域及成员函数的声明与定义分离。同时深入探讨了类的大小计算、`this`指针、默认成员函数(构造函数、析构函数、拷贝构造、赋值重载)以及运算符重载等内容。 文章还详细分析了`explicit`关键字的作用、静态成员(变量与函数)、友元(友元函数与友元类)的概念及其使用场景,并简要介绍了内部类的特性。
424 0