【C++】学习笔记——类和对象_3

embedded/2024/9/23 9:24:12/

文章目录

  • 二、类和对象
    • 11. 析构函数(补)
    • 12. 拷贝构造函数
    • 13. 运算符重载
  • 未完待续


二、类和对象

11. 析构函数(补)

析构函数并不是销毁对象,对象的销毁是由编译器完成的,析构函数的作用是清理,清理对象在堆上占用的空间
析构函数的顺序:

#include<iostream>
using namespace std;class A
{
public:A(char a){_a = a;cout << "构造函数->" << _a << endl;}~A(){cout << "析构函数->" << _a << endl;}
private:char _a;
};A e('e');void func()
{A c('c');static A d('d');
}static A f('f');int main()
{A a('a');A b('b');func();return 0;
}

结果:
在这里插入图片描述
看生命周期,生命周期先结束的先析构,生命周期同时结束的,先构造的后析构。注意,局部的静态对象虽然声明周期属于全局,但是相比于正宗的全局对象,局部的静态对象先析构。

12. 拷贝构造函数

拷贝构造简单说就是,构造一个对象,但是这个对象的值是拷贝另一个已存在的对象的。所以拷贝构造函数的参数就是一个对象。

#include<iostream>
using namespace std;class Date
{
public:// 构造函数Date(int year = 1111, int month = 1, int day = 1){_year = year;_month = month;_day = day;}// 析构函数~Date(){// 空}// 拷贝构造函数Date(const Date d){_year = d._year;_month = d._month;_day = d._day;}void Print(){cout << _year << "->" << _month << "->" << _day << endl;}
private:int _year;int _month;int _day;
};int main()
{Date d1(2222, 2, 2);Date d2(d1);d1.Print();d2.Print();return 0;
}

就像这样,但是上面写法是不对的,会报错。
在这里插入图片描述
这是为什么呢?因为在C++当中,如果将一个自定义类型(结构体或者类)去传值传参时,由于形参是实参的临时拷贝,所以相当于形参要拷贝一份实参,拷贝拷贝,形参就是调用拷贝构造函数了。拷贝构造函数也是函数,如果形参也是传值传参时,拷贝构造函数就要去调用一个新的拷贝构造函数,无穷递归矣。所以拷贝构造函数的参数不能是传值传参,要使用引用传参
使用了引用传参,注意要加上const哦,保护被拷贝的那个对象。
在这里插入图片描述
拷贝构造函数也是默认成员函数,我们不显式定义,编译器会自动生成。不过,它与构造函数和析构函数不一样,它对内置类型进行浅拷贝(也叫值拷贝,就是按字节一个一个拷贝),对自定义类型调用自定义类型的拷贝构造函数。咦?自定义类型最终也是属于内置类型,所以说,拷贝构造函数是不是不用写,编译器自动生成就很完美了呢?
当然不是,这涉及到了浅拷贝和深拷贝的问题了。让我们看下面一段程序:

#include<iostream>
using namespace std;typedef int DataType;class Stack
{
public:// 初始化栈Stack(size_t capacity = 10){_array = (DataType*)malloc(capacity * sizeof(DataType));if (nullptr == _array){perror("malloc申请空间失败");return;}_size = 0;_capacity = capacity;}// 入栈void Push(const DataType& data){// CheckCapacity();_array[_size] = data;_size++;}// 析构~Stack(){if (_array){free(_array);_array = nullptr;_capacity = 0;_size = 0;}}
private:DataType* _array;size_t _size;size_t _capacity;
};int main()
{Stack s1;s1.Push(1);s1.Push(2);s1.Push(3);s1.Push(4);Stack s2(s1);return 0;
}

这个程序语法上没问题,但是编译出错了。为啥呢?这是s2拷贝完s1后的结果:
在这里插入图片描述
不知道大家有没有发现错误,错误就是,由于拷贝构造是浅拷贝的问题,按字节一个一个拷贝,于是指针的指向也被拷贝过去了,这使得两个指针指向同一块空间,s2先析构,析构会导致空间被释放,轮到s1析构的时候,该空间已被释放过了,再次释放就产生了错误。所以说:类中如果没有涉及资源申请时,拷贝构造函数是否写都可以;一旦涉及到资源申请时,则拷贝构造函数是一定要写的,否则就是浅拷贝

拷贝构造函数也是特殊的成员函数,其特征如下:

  1. 拷贝构造函数是构造函数的一个重载形式。
  2. 拷贝构造函数的参数只有一个且必须是类类型对象的引用,使用传值方式编译器直接报错,因为会引发无穷递归调用。
  3. 若未显式定义,编译器会生成默认的拷贝构造函数。 默认的拷贝构造函数对象按内存存储按字节序完成拷贝,这种拷贝叫做浅拷贝,或者值拷贝。

13. 运算符重载

C++为了增强代码的可读性引入了运算符重载,运算符重载是具有特殊函数名的函数,也具有其返回值类型,函数名字以及参数列表,其返回值类型与参数列表与普通的函数类似。函数名字为:关键字operator后面接需要重载的运算符符号。
函数原型:返回值类型 operator操作符(参数列表)

#include<iostream>
using namespace std;class Date
{
public:Date(int year = 1111, int month = 1, int day = 1){_year = year;_month = month;_day = day;}~Date(){}// 重载==bool operator==(const Date& d){return _year == d._year&& _month == d._month&& _day == d._day;}
private:int _year;int _month;int _day;
};int main()
{Date d1(2222, 2, 2);Date d2(2222, 2, 2);cout << d1.operator==(d2) << endl;// 也可以写成这样,由于流插入运算符优先级较高,所以要加上()cout << (d1 == d2) << endl;return 0;
}

在这里插入图片描述

注意:
不能通过连接其他符号来创建新的操作符:比如operator@。
重载操作符必须有一个类类型参数。
用于内置类型的运算符,其含义不能改变,例如:内置的整型+,不能改变其含义。
作为类成员函数重载时,其形参看起来比操作数数目少1,因为成员函数的第一个参数为隐藏的this指针。
.*, :: , sizeof , ?:,. 注意以上5个运算符不能重载。


未完待续


http://www.ppmy.cn/embedded/16066.html

相关文章

Sping源码(七)—context: component-scan标签如何扫描、加载Bean

序言 简单回顾一下。上一篇文章介绍了从xml文件context component-scan标签的加载流程到ConfigurationClassPostProcessor的创建流程。 本篇会深入了解context component-scan标签底层做了些什么。 component-scan 早期使用Spring进行开发时&#xff0c;很多时候都是注解 标…

记录如何用php删除一段文字中的指定文字

在PHP中&#xff0c;删除一段文字中的指定文字可以通过使用str_replace()函数来实现。这个函数可以替换字符串中的某些字符或字符串&#xff0c;如果你想要删除特定的文字&#xff0c;可以将它们替换为空字符串。 以下是使用str_replace()函数删除指定文字的基本语法&#xff…

Spring 数据脱敏实现方式

1、前言 当前互联网中&#xff0c;越来越重视数据安全&#xff0c;数据脱敏在实际应用中越来越多。 2 、脱敏方式 2.1 数据库sql 语句脱敏 sql 语句脱敏是比较传统通用的&#xff0c;例子如下所示&#xff1a; select CONCAT(LETF(mobile,3),"*****",RIGHT(mobile,…

jupyter notebook用不了multiporcessing的问题

在跑一本drl书中的代码遇到这个问题&#xff0c;到书的github页面看到别人也有类似的问题 搬运答案 Had the same issue, this is because multiprocessing does not work natively inside a Jupyter notebook. For details see this StackOverflow discussion (specifically…

Vision Pro“裸眼上车”,商汤绝影全新舱内3D交互亮相

2023年&#xff0c;Apple Vision Pro的横空出世让人们领略到了3D交互的魅力&#xff0c;商汤绝影通过深厚的技术研发实力和高效的创新迭代效率&#xff0c;带来两大全新座舱3D交互&#xff1a;3D Gaze高精视线交互和3D动态手势交互。 作为全球首创的能够通过视线定位与屏幕图标…

Mybatis二级缓存

如何开启二级缓存 1.在配置文件中&#xff0c;将cacheEnabled属性设置为true &#xff08;默认true&#xff09; <settings><setting name"cacheEnabled" value"true"/> </settings> 2.声明缓存空间 2.1 在xml文件中添加<cache/&…

【Python】异常、模块与包

目录 捕获异常 异常的传递 Python中的模块 模块的导入方式 as定义别名 自定义模块 Python包 第三方包 综合案例 当我们的程序遇到了BUG, 那么接下来有两种情况: ① 整个程序因为一个BUG停止运行 ② 对BUG进行提醒, 整个程序继续运行 但是在真实工作中, 我们肯定不能…

Flink容错机制

Flink的容错机制是一个复杂而精细的系统&#xff0c;旨在确保在分布式流处理过程中&#xff0c;即使在发生故障的情况下&#xff0c;也能保持数据的一致性和计算的正确性。以下是对Flink容错机制的详细阐述&#xff1a; 首先&#xff0c;Flink的容错机制建立在状态一致性的基础…