从C语言到C++⑤(第二章_类和对象_中篇)(6个默认成员函数+运算符重载+const成员)(上)

简介: 从C语言到C++⑤(第二章_类和对象_中篇)(6个默认成员函数+运算符重载+const成员)

0. 引入6个默认成员函数

如果一个类中什么成员都没有,简称为空类。

空类中真的什么都没有吗?并不是,任何类在什么都不写时,

编译器会自动生成以下 6 个默认成员函数。

C++类中有6个默认函数,分别是:

构造函数、 析构函数、 拷贝构造函数、 赋值运算符重载、 取地址及 const取地址运算符重载。

这六个函数是很特殊的函数,如果我们不自己实现,编译器就会自己实现。

默认成员函数:用户没有显式实现,编译器会生成的成员函数称为默认成员函数。

对于默认成员函数,如果我们不主动实现,编译器会自己生成一份。

比如我们在上一篇里举过的一个 Stack 的例子,如果需要初始化和清理,"构造函数" 和 "析构函数" 就可以帮助我们完成。构造函数就类似于 Init,而析构函数就类似于 Destroy。

1. 构造函数(默认成员函数)

对于以下 Date

#include <iostream>
using namespace std;
 
class Date 
{
public:
    void Init(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
    void Print()
    {
        printf("%d-%d-%d\n", _year, _month, _day);
    }
 
private:
    int _year;
    int _month;
    int _day;
};
 
int main()
{
    Date d1;
    d1.Init(2023, 4, 23);
    d1.Print();
 
    Date d2;
    d2.Init(2022, 5, 2);
    d2.Print();
 
    return 0;
}

       对于Date 类,可以通过 Init 公有方法给对象设置日期,但如果每次创建对象时都调用该方法设置信息,未免有点麻烦,而且写其它类的时候可能会忘记初始化,会出现程序崩溃的情况, 那能否在对象创建时,就将信息设置进去呢?

1.1 构造函数的概念

构造函数是一个特殊的成员函数,名字与类名相同,创建类类型对象时由编译器自动调用,

以保证每个数据成员都有一个合适的初始值,并且在对象整个生命周期内只调用一次。

构造函数的意义:能够保证对象被初始化。

构造函数是特殊的成员函数,主要任务是初始化对象,而不是开空间。

(虽然构造函数的名字叫构造)


1.2 构造函数的特性和用法

构造函数是特殊的成员函数,主要特征如下:

① 构造函数的函数名和类名是相同的

② 构造函数无返回值(也不用写void)

③ 构造函数可以重载

④ 会在对象实例化时自动调用对象定义出来。

构造函数的用法:

#include <iostream>
using namespace std;
 
class Date 
{
public:
    Date()
    {
        _year = 1;
        _month = 1;
        _day = 1;
    }
 
    Date(int year, int month, int day)
    {
        _year = year;
        _month = month;
        _day = day;
    }
 
    void Print()
    {
        printf("%d-%d-%d\n", _year, _month, _day);
    }
 
private:
    int _year;
    int _month;
    int _day;
};
 
int main()
{
    Date d1; // 对象实例化,此时触发构造,调用无参构造函数
    d1.Print();
 
    Date d2(2023, 5, 2); // 对象实例化,此时触发构造,调用带参构造函数
    // 这里如果调用带参构造函数,我们需要传递三个参数(这里我们没设缺省) 。
    //如果想传几个就传几个可以自己设置重载
    d2.Print();
 
    // 注意:如果通过无参构造函数创建对象时,对象后面不用跟括号,否则就成了函数声明
    // 以下代码的函数:声明了d3函数,该函数无参,返回一个日期类型的对象
    Date d3();
    // warning C4930: “Date d3(void)”: 未调用原型函数(是否是有意用变量定义的?)
 
    //构造函数是特殊的,不是常规的成员函数,不能直接调d1.Data();
 
    return 0;
}

       如果类中没有显式定义构造函数,则C++编译器会自动生成一个无参的默认构造函数,

一旦用户显式定义编译器将不再生成。

       对于上面的d1中,如果只有带参构造函数就会报错,对于d2,如果只有无参构造函数就会报错,所以把自己写的构造函数都删除之后d1可以运行,d2会报错。


1.3 默认构造函数

class Date 
{
public:
    //无参构造函数 是 默认构造函数 
    Date()
    {
        _year = 1;
        _month = 1;
        _day = 1;
    }
 
    //全缺省构造函数 也是 默认构造函数 (一般写全缺省,不写上面那个)
    Date(int year = 1, int month = 1, int day = 1) 
    {
        _year = year;
        _month = month;
        _day = day;
    }
 
private:
    int _year;
    int _month;
    int _day;
};

无参构造函数、全缺省构造函数都被称为默认构造函数。并且默认构造函数只能有一个。

注意事项:

       ① 无参构造函数、全缺省构造函数、我们没写编译器默认生成的无参构造函数,这三个都可以认为是默认构造函数。

       ② 语法上无参和全缺省可以同时存在,但如果同时存在会引发二义性:无参的构造函数和全缺省的构造函数都成为默认构造函数,并且默认构造参数只能有一个,语法上他们两个可以同时存在,但是如果有对象定义去调用就会报错。

       关于编译器生成的默认成员函数,很多人会有疑惑:不实现构造函数的情况下,编译器会生成默认的构造函数。但是看起来默认构造函数又没什么用?d 对象调用了编译器生成的默认构造函数,但是d 对象 _year/_month/_day ,依旧是随机值。也就说在这里编译器生成的 默认构造函数并没有什么用?? 解答:

C++把类型分成 内置类型(基本类型)和自定义类型

内置类型就是语言提供的数据类型,如:int/char/指针等等,

自定义类型就是我们使用class/struct等自己定义的类型,

C++ 规定:我们不写编译器默认生成构造函数,对于内置类型的成员变量,不做初始化处理。

但是对于自定义类型的成员变量会去调用它的默认构造函数(不用参数就可以调的)初始化。

如果没有默认构造函数(不用参数就可以调用的构造函数)就会报错。

#include <iostream>
using namespace std;
class Time
{
public:
  Time()
  {
    cout << "Time()" << endl;
    _hour = 0;
    _minute = 0;
    _second = 0;
  }
private:
  int _hour;
  int _minute;
  int _second;
};
 
class Date
{
public:
 
  void Print()
  {
    printf("%d %d %d\n", _year, _month, _day);
  }
 
private:
 
  int _year;// 基本类型(内置类型)
  int _month;
  int _day;
 
  Time _t;// 自定义类型
};
 
int main()
{
  Date d;
  d.Print();
  return 0;
}

       很多人吐槽不写构造函数编译器会默认生成的这个特性设计得不好,因为没有对内置类型和自定义类型统一处理,不处理内置类型成员变量,只处理自定义类型成员变量。

       但是覆水难收,所以C++11 中针对内置类型成员不初始化的缺陷,又打了补丁:内置类型成员变量在类中声明时可以给默认值:

#include <iostream>
using namespace std;
class Time
{
public:
  Time()
  {
    cout << "Time()" << endl;
    _hour = 0;
    _minute = 0;
    _second = 0;
  }
private:
  int _hour;
  int _minute;
  int _second;
};
 
class Date
{
public:
 
  void Print()
  {
    printf("%d %d %d\n", _year, _month, _day);
  }
 
private:
  
  int _year = 1;// 基本类型(内置类型)
  int _month = 1;
  int _day = 1;
  //注意这里不是初始化,是给默认构造函数缺省值
 
  Time _t;// 自定义类型
};
 
int main()
{
  Date d;
  d.Print();
  return 0;
}

需要注意的是,上面代码中如果自定义类型Time没有写构造函数,编译器也什么都不会处理。

总结:

构造函数分为三类:

①无参构造函数、

②全缺省构造函数、

③我们没写编译器默认生成的构造函数,

       这三类都可以认为是默认构造函数。并且默认构造函数只能有一个。 一般的类都不会让编译器默认生成构造函数,一般显示地写一个全缺省,非常好用, 特殊情况才会默认生成。

2. 析构函数(默认成员函数)

2.1 析构函数概念

       通过前面构造函数的学习,我们知道一个对象是怎么来的,那一个对象又是怎么没呢的?

       析构函数:与构造函数功能相反,析构函数不是完成对对象本身的销毁,局部对象销毁工作是由编译器完成的。而对象在销毁时会 自动调用 析构函数,完成对象中资源的清理工作。

以前我们写数据结构的时候经常忘记调用 destroy 函数,但是现在我们有析构函数了。

2.2 析构函数特性

构造函数是特殊的成员函数,主要特征如下:

① 析构函数名是在类名前面加上字符

② 析构函数既没有参数也没有返回值(因为没有参数,所以也不会构成重载问题)

③ 一个类的析构函数有且仅有一个(如果不写系统会默认生成一个析构函数)

④ 析构函数在对象生命周期结束后,会自动调用。(和析构函数是对应的构造函数是在对象实例化时自动调用)

#include <iostream>
using namespace std;
 
class Date 
{
public:
    Date(int year = 1, int month = 1, int day = 1) 
    {
        _year = year;
        _month = month;
        _day = day;
    }
    void Print() 
    {
        printf("%d-%d-%d\n", _year, _month, _day);
    }
 
    ~Date() 
    {
        cout << "~Date()" << endl;// 日期类没有资源需要清理,所以只打印下知道调用了
    }
 
private:
    int _year;
    int _month;
    int _day;
};
 
int main()
{
    Date d1;
    Date d2(2023, 5, 2);
 
    return 0;
}

d1 和 d2 都会调用析构函数:

       拿 Stack 来举个例子,体会下构造函数和析构函数的用处,我们知道,栈是需要 destroy 清理开辟的内存空间的。

#include<iostream>
#include<stdlib.h>
using namespace std;
 
typedef int StackDataType;
class Stack 
{
public:
    Stack(int capacity = 4) // 这里只需要一个capacity就够了,默认给4(利用缺省参数)
    {
        _array = (StackDataType*)malloc(sizeof(StackDataType) * capacity);
        if (_array == NULL) 
        {
            cout << "Malloc Failed!" << endl;
            exit(-1);
        }
        _top = 0;
        _capacity = capacity;
    }
 
    ~Stack() // 这里就用的上析构函数了,我们需要清理开辟的内存空间(防止内存泄漏)
    {
        free(_array);
        _array = nullptr;//下面这两行可以不写,这个野指针已经没人能访问到了
        _top = _capacity = 0;//但写了也是个好习惯
    }
 
private:
    int* _array;
    size_t _top;
    size_t _capacity;
};
 
int main(void)
{
    Stack s1;
    Stack s2(20); //初始capacity给20
 
    return 0;
}

       代码解读:我们在设置栈的构造函数时,定义容量 capacity 时利用缺省参数默认给个4的容量,这样用的时候默认就是4,如果不想要4可以自己传。如此一来,就可以保证了栈被定义出来就一定被初始化,用完后会自动销毁。以后就不会有忘记调用 destroy 而导致内存泄露的惨案了,这里的析构函数就可以充当销毁的作用。

       如果类中没有申请资源时,析构函数可以不写,直接使用编译器生成的默认析构函数,比如 Date类;有资源申请时,一定要写,否则会造成资源泄漏,比如Stack类。

       有没有想过,这里是先析构 s1 还是先析构 s2?既然都这样问了,应该是先析构 s2 了 ,没错没错,栈帧和栈里面的对象都符合栈的性质,析构的顺序在局部的栈中是相反的,栈帧销毁清理资源时 s2 先清理,然后再清理 s1 。(可以在析构函数打印参数看看)(贴两个图)

3f5bea08a11343cfb0c99e069f559a32.png

这张图3也是全局的:

       如果我们不自己写析构函数,让编译器自动生成,那么这个默认析构函数和默认构造函数类似: ① 对于 "内置类型" 的成员变量:不作处理,② 对于 "自定义类型" 的成员变量:会调用它对应的析构函数。

       可能有人要说帮我都销毁掉不就好了?举个最简单的例子,迭代器,析构的时候是不释放的,因为不需要析构函数来管,所以默认不对内置类型处理是正常的,这么一来默认生成的析构函数不就没有用了吗?

       有用,他对内置类型的成员类型不作处理,会在一些情况下非常的有用。

从C语言到C++⑤(第二章_类和对象_中篇)(6个默认成员函数+运算符重载+const成员)(中):https://developer.aliyun.com/article/1513647?spm=a2c6h.13148508.setting.14.5e0d4f0eApSShM

目录
相关文章
|
2天前
|
编译器 C语言 C++
类和对象的简述(c++篇)
类和对象的简述(c++篇)
|
1月前
|
C++ 芯片
【C++面向对象——类与对象】Computer类(头歌实践教学平台习题)【合集】
声明一个简单的Computer类,含有数据成员芯片(cpu)、内存(ram)、光驱(cdrom)等等,以及两个公有成员函数run、stop。只能在类的内部访问。这是一种数据隐藏的机制,用于保护类的数据不被外部随意修改。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。成员可以在派生类(继承该类的子类)中访问。成员,在类的外部不能直接访问。可以在类的外部直接访问。为了完成本关任务,你需要掌握。
68 19
|
1月前
|
存储 编译器 数据安全/隐私保护
【C++面向对象——类与对象】CPU类(头歌实践教学平台习题)【合集】
声明一个CPU类,包含等级(rank)、频率(frequency)、电压(voltage)等属性,以及两个公有成员函数run、stop。根据提示,在右侧编辑器补充代码,平台会对你编写的代码进行测试。​ 相关知识 类的声明和使用。 类的声明和对象的声明。 构造函数和析构函数的执行。 一、类的声明和使用 1.类的声明基础 在C++中,类是创建对象的蓝图。类的声明定义了类的成员,包括数据成员(变量)和成员函数(方法)。一个简单的类声明示例如下: classMyClass{ public: int
50 13
|
1月前
|
编译器 数据安全/隐私保护 C++
【C++面向对象——继承与派生】派生类的应用(头歌实践教学平台习题)【合集】
本实验旨在学习类的继承关系、不同继承方式下的访问控制及利用虚基类解决二义性问题。主要内容包括: 1. **类的继承关系基础概念**:介绍继承的定义及声明派生类的语法。 2. **不同继承方式下对基类成员的访问控制**:详细说明`public`、`private`和`protected`继承方式对基类成员的访问权限影响。 3. **利用虚基类解决二义性问题**:解释多继承中可能出现的二义性及其解决方案——虚基类。 实验任务要求从`people`类派生出`student`、`teacher`、`graduate`和`TA`类,添加特定属性并测试这些类的功能。最终通过创建教师和助教实例,验证代码
52 5
|
1月前
|
存储 算法 搜索推荐
【C++面向对象——群体类和群体数据的组织】实现含排序功能的数组类(头歌实践教学平台习题)【合集】
1. **相关排序和查找算法的原理**:介绍直接插入排序、直接选择排序、冒泡排序和顺序查找的基本原理及其实现代码。 2. **C++ 类与成员函数的定义**:讲解如何定义`Array`类,包括类的声明和实现,以及成员函数的定义与调用。 3. **数组作为类的成员变量的处理**:探讨内存管理和正确访问数组元素的方法,确保在类中正确使用动态分配的数组。 4. **函数参数传递与返回值处理**:解释排序和查找函数的参数传递方式及返回值处理,确保函数功能正确实现。 通过掌握这些知识,可以顺利地将排序和查找算法封装到`Array`类中,并进行测试验证。编程要求是在右侧编辑器补充代码以实现三种排序算法
40 5
|
1月前
|
Serverless 编译器 C++
【C++面向对象——类的多态性与虚函数】计算图像面积(头歌实践教学平台习题)【合集】
本任务要求设计一个矩形类、圆形类和图形基类,计算并输出相应图形面积。相关知识点包括纯虚函数和抽象类的使用。 **目录:** - 任务描述 - 相关知识 - 纯虚函数 - 特点 - 使用场景 - 作用 - 注意事项 - 相关概念对比 - 抽象类的使用 - 定义与概念 - 使用场景 - 编程要求 - 测试说明 - 通关代码 - 测试结果 **任务概述:** 1. **图形基类(Shape)**:包含纯虚函数 `void PrintArea()`。 2. **矩形类(Rectangle)**:继承 Shape 类,重写 `Print
48 4
|
1月前
|
设计模式 IDE 编译器
【C++面向对象——类的多态性与虚函数】编写教学游戏:认识动物(头歌实践教学平台习题)【合集】
本项目旨在通过C++编程实现一个教学游戏,帮助小朋友认识动物。程序设计了一个动物园场景,包含Dog、Bird和Frog三种动物。每个动物都有move和shout行为,用于展示其特征。游戏随机挑选10个动物,前5个供学习,后5个用于测试。使用虚函数和多态实现不同动物的行为,确保代码灵活扩展。此外,通过typeid获取对象类型,并利用strstr辅助判断类型。相关头文件如&lt;string&gt;、&lt;cstdlib&gt;等确保程序正常运行。最终,根据小朋友的回答计算得分,提供互动学习体验。 - **任务描述**:编写教学游戏,随机挑选10个动物进行展示与测试。 - **类设计**:基类
32 3
|
2月前
|
安全 程序员 编译器
【C语言】const 关键字详解
`const`关键字在C语言中用于定义常量,提供只读的变量。这意味着一旦初始化,`const`变量的值不能再被修改。下面详细介绍`const`关键字的用法、作用以及其在不同上下文中的应用。
87 2
|
3月前
|
存储 编译器 C语言
【c++丨STL】string类的使用
本文介绍了C++中`string`类的基本概念及其主要接口。`string`类在C++标准库中扮演着重要角色,它提供了比C语言中字符串处理函数更丰富、安全和便捷的功能。文章详细讲解了`string`类的构造函数、赋值运算符、容量管理接口、元素访问及遍历方法、字符串修改操作、字符串运算接口、常量成员和非成员函数等内容。通过实例演示了如何使用这些接口进行字符串的创建、修改、查找和比较等操作,帮助读者更好地理解和掌握`string`类的应用。
89 2
|
3月前
|
存储 编译器 C++
【c++】类和对象(下)(取地址运算符重载、深究构造函数、类型转换、static修饰成员、友元、内部类、匿名对象)
本文介绍了C++中类和对象的高级特性,包括取地址运算符重载、构造函数的初始化列表、类型转换、static修饰成员、友元、内部类及匿名对象等内容。文章详细解释了每个概念的使用方法和注意事项,帮助读者深入了解C++面向对象编程的核心机制。
156 5