• C++基础——赋值运算符重载函数


    前言:

            上篇博客中,我们学习了运算符重载函数的定义、格式与特征,介绍了众多运算符的重载编 写过程:C++基础——运算符重载函数

    这次我们来学习一个新的运算符重载函——赋值运算符重载函数。

    说一下运算符重载函数的由来:

           运算符重载函数是用来体现运算符的多态性,在学习C语言的过程中,运算符全都是用来解决内置类型数据、逻辑、关系表达式间的运算,而在学习C++的过程中,我们了解了类与对象,就会思考到自定义类型的对象之间是否也能进行+-*/等多种运算呢?基于此种想法,C++官方研发出了运算符重载函数,就是为了让自定义类型也能做运算,体现出了语言的多态性。

     


    目录

    前言

    一.赋值运算符重载

    练习:

    二.默认运算符重载函数

    练习:


    一.赋值运算符重载

            赋值运算符重载函数和拷贝构造有着很相似的地方,都是将一个对象的数据拷贝赋值给另一个对象。但不同点在于:

    拷贝构造函数是一个拷贝初始化另一个马上要创建的对象;

    而赋值运算符重载是两个对象都已经创建好后,进行拷贝赋值。

    1. Date d0(1500, 3, 17);
    2. //拷贝构造
    3. Date d00(d0);
    4. Date d01 = d0; //也属于拷贝构造,它等价于 Date d01(d0);
    5. d01 = d0; //赋值重载

    练习:

    1. //赋值运算符重载
    2. class Date {
    3. public:
    4. Date(int year = 1199, int month = 12, int day = 15) {
    5. _year = year;
    6. _month = month;
    7. _day = day;
    8. }
    9. void operator=(const Date& d) {
    10. _year = d._year;
    11. _month = d._month;
    12. _day = d._day;
    13. }
    14. void Print() {
    15. cout << _year << "-" << _month << "-" << _day << endl;
    16. }
    17. private:
    18. int _year;
    19. int _month;
    20. int _day;
    21. };
    22. int main() {
    23. Date d1(2022, 10, 19);
    24. Date d2(2021, 5, 17);
    25. d1.Print();
    26. d1 = d2; //调用operator=赋值运算符重载函数
    27. d1.Print();
    28. return 0;
    29. }

    void operator=这种写法只能支持单次赋值操作,而我们之前应该写过:

    int a=10,b=20,c=50;

    a=b=c;         

            这种赋值方式称为连续赋值——链式访问,它是通过将c值传给b,然后其返回值再传给a,我们当前的函数没有返回值,所以它只能是一次性的赋值,若想要变成连续赋值函数,需要:

    1. //支持连续赋值
    2. Date& operator=(const Date& d) {
    3. _year = d._year;
    4. _month = d._month;
    5. _day = d._day;
    6. return *this;
    7. }

    解析:使用引用返回的目的是为了减少拷贝次数,提高运行效率,对于引用返回不太明白的小伙伴可以去看看: C++基础——引用讲解2

     里面介绍了传值返回和引用返回的区别。

            return *this时,this是局部指针,出了作用域会被销毁,但*this不会,*this是对象d1,d1的生命周期是全局性的,所以可以使用引用返回。

     

    二.默认运算符重载函数

            当我们不写赋值重载函数时,编译器会自动生成一个默认的赋值重载函数,它的运算方式是按照字节序的方式逐步完成拷贝。

    练习:

    1. class Date {
    2. public:
    3. Date(int year = 1199, int month = 12, int day = 15) {
    4. _year = year;
    5. _month = month;
    6. _day = day;
    7. }
    8. //没有写赋值重载函数
    9. void Print() {
    10. cout << _year << "-" << _month << "-" << _day << endl;
    11. }
    12. private:
    13. int _year;
    14. int _month;
    15. int _day;
    16. };
    17. int main() {
    18. Date d1(2022, 10, 30);
    19. Date d2(1000, 1, 1);
    20. Date d3(2090, 12, 10);
    21. cout << "d1更改前:" << endl;
    22. d1.Print();
    23. d1 = d2;
    24. cout << "d1更改后:" << endl;
    25. d1.Print();
    26. cout << "d2更改前:" << endl;
    27. d2.Print();
    28. d2 = d1 = d3;
    29. cout << "d2更改后:" << endl;
    30. d2.Print();
    31. return 0;
    32. }

            注:内置类型成员变量是以字节序的方式直接赋值,而自定义类型成员变量是需要调用对应类的赋值重载函数完成赋值。

            所以日期类的赋值重载并不需要亲自去写。

    需要自己写赋值重载的类为:

    1. //需要自己写赋值的有Stack类:
    2. typedef int DataType;
    3. class Stack{
    4. public:
    5. Stack(size_t capacity = 10){
    6. _array = (DataType*)malloc(capacity * sizeof(DataType));
    7. if (nullptr == _array){
    8. perror("malloc申请空间失败");
    9. return;
    10. }
    11. _size = 0;
    12. _capacity = capacity;
    13. }
    14. void Push(const DataType& data){
    15. // CheckCapacity();
    16. _array[_size] = data;
    17. _size++;
    18. }
    19. ~Stack(){
    20. if (_array){
    21. free(_array);
    22. _array = nullptr;
    23. _capacity = 0;
    24. _size = 0;
    25. }
    26. }
    27. private:
    28. DataType* _array;
    29. size_t _size;
    30. size_t _capacity;
    31. };
    32. int main(){
    33. Stack s1;
    34. s1.Push(1);
    35. s1.Push(2);
    36. s1.Push(3);
    37. s1.Push(4);
    38. Stack s2;
    39. s1 = s2; //赋值重载
    40. return 0;
    41. }

            若我们使用默认的赋值重载函数,会报错,原因与之前拷贝构造中Stack类的错误原因相同,都是对同一块空间析构了两次,不仅如此,还存在内存泄漏。

            针对这两个错误,我们需要自己去写赋值重载函数。若想不造成内存泄漏和析构错误,那么我们需要把左值对象的_array所指向的空间给提前释放掉,然后再重新malloc一个与被拷贝对象空间同等大小的空间,这样一来,两个问题都会被解决。

    1. //赋值运算符重载函数
    2. Stack& operator=(const Stack& st) {
    3. free(_array);
    4. _array = (DataType*)malloc(sizeof(DataType) * st._capacity);
    5. if (_array == nullptr) {
    6. perror("malloc fail");
    7. exit(-1);
    8. }
    9. memcpy(_array, st._array, sizeof(DataType) * st._size);
    10. _size = st._size;
    11. _capacity = st._capacity;
    12. }

     ​​​​

  • 相关阅读:
    SPA项目之主页面--数据表格的增删改查
    流程图渲染方式:Canvas vs SVG
    20.3 OpenSSL 对称AES加解密算法
    《Redis基础篇》带你走进Redis的世界 ~ ⭐必看必看⭐
    计算机毕业设计Python+Django的银行取号排队系统
    jvm 内存结构 ^_^
    数字藏品实物化金谷诺亚艺博馆布局 携手翰德轩转变多维创新
    快速分割视频,并提取原视频中的音频单独保存
    ardupilot避障代码分析
    vue中的常用指令
  • 原文地址:https://blog.csdn.net/weixin_69283129/article/details/127792440