C++运算符重载(三)之递增运算符重载

简介: 如果返回值是引用,局部对象在当前成员函数执行完后释放,再返回局部对象的引用就是非法操作。如果返回值是一个值,实际上返回的是一个值的副本,因为返回是一个拷贝构造过程,原来的释放了,但是拷贝了一份新的,不受成员函数释放的影响。(详细见前面文章回顾)这里可以直接 new 一个类,然后就可以返回引用了。



递增运算符重载

作用: 通过重载递增运算符,实现自己的整型数据

重载前置递增运算符

classMyInteger {

   friendostream&operator<<(ostream&out, MyIntegermyint);

public:

   MyInteger() {

       m_Num=0;

   }

   //前置++

   MyInteger&operator++() {//注意&

       //先++

       m_Num++;

       //再返回

       return*this;

   }

private:

   intm_Num;

};

//左移运算符重载

ostream&operator<<(ostream&out, MyIntegermyint) {//加上&就是引用传递。

   out<<myint.m_Num;

   returnout;

}

//前置++ 先++ 再返回

voidtest01() {

   MyIntegermyInt;

   cout<<++myInt<<endl;//先++,后输出

   cout<<myInt<<endl;

}

intmain() {

   test01();//输出结果:1 1

   system("pause");

   return0;

}

注意:cout << ++myInt << endl;//先++,后输出

先++,后输出是指先运行成员函数前置递增运算符重载,再运行函数左移运算符重载。所以先++完成后再传入左移运算符重载函数中,要么是引用传递,要么是拷贝传递,上图使用的是拷贝传递,都可。

PS:为什么MyInteger& operator++() {}处要使用&

//预期目的:两次递增运算都是作用在同一个对象上

inta=0;

cout<<++(++a) <<endl;//2

//如果返回值是引用,那么返回值就是本身,如果返回值是一个值,实际上返回的是一个值的副本,拷贝构造。

//若是去除了引用,拷贝构造函数被调用,创建了临时对象,没有在原对象上进行操作,所以输出的不一样。

//引用是为了对一个数据进行递增操作

MyInteger&operator++() {

   m_Num++;

   return*this;

}

重载后置递增运算符

#include<iostream>

usingnamespacestd;

classMyInteger {

   friendostream&operator<<(ostream&out, MyIntegermyint);//注意&

public:

   MyInteger() {

       m_Num=0;

   }

   //后置++ ,int代表占位参数,可以用于区分前置和后置递增。

   MyIntegeroperator++(int) {

       //先返回

       MyIntegertemp=*this; //记录当前本身的值,然后让本身的值加1,但是返回的是以前的值,达到先返回后++;

       m_Num++;

       returntemp;//拷贝构造

   }

private:

   intm_Num;

};

//左移运算符重载

ostream&operator<<(ostream&out, MyIntegermyint) {

   out<<myint.m_Num;

   returnout;

}

//后置++ 先返回 再++

voidtest02() {

   MyIntegermyInt;

   cout<<myInt++<<endl;//先输出后++

   cout<<myInt<<endl;

}

intmain() {

   

   test02();//输出结果:0 1

   system("pause");

   return0;

}

PS:后置递增返回的原因:

   MyIntegeroperator++(int) {

       //先返回

       MyIntegertemp=*this; //记录当前本身的值,然后让本身的值加1,但是返回的是以前的值,达到先返回后++;

       m_Num++;

       returntemp;

   }

如果返回值是引用,局部对象在当前成员函数执行完后释放,再返回局部对象的引用就是非法操作。如果返回值是一个值,实际上返回的是一个值的副本,因为返回是一个拷贝构造过程,原来的释放了,但是拷贝了一份新的,不受成员函数释放的影响。(详细见前面文章回顾)

这里可以直接 new 一个类,然后就可以返回引用了。

#include<iostream>

usingnamespacestd;

classMyInteger {

   friendostream&operator<<(ostream&out, MyInteger&myint);

public:

   MyInteger() {

       m_Num=0;

   }

   //后置++ ,int代表占位参数,可以用于区分前置和后置递增。

   //在堆区创建

   MyInteger&operator++(int) {

       temp=newMyInteger(*this);

       m_Num++;

       return*temp;

   }

   ~MyInteger() {//析构时释放堆区

       if (temp!=NULL)

       {

           deletetemp;

           temp=NULL;

       }

       

   }

public:

   MyInteger*temp ;

private:

   intm_Num;

};

//左移运算符重载

ostream&operator<<(ostream&out, MyInteger&myint) {//注意此处是取址符

   out<<myint.m_Num;

   returnout;

}

//后置++ 先返回 再++

voidtest02() {

   MyIntegermyInt;

   cout<<myInt++<<endl;//先输出,后++

   cout<<myInt<<endl;

}

intmain() {

   test02();//输出结果:0 1

   system("pause");

   return0;

}

注意1cout << myInt++ << endl;//先输出,后++

虽然是先输出后++,但是运行时同前置递增重载运行顺序,先运行后置递增重载成员函数,再运行左移运算符重载全局函数。

cout << myInt++ << endl;//先输出,后++先调用 左移运算符重载全局函数 输出开辟到堆区的*temp再对栈区的myInt做后置++操作

cout << myInt << endl;之后再执行第二次输出,再次调用 左移运算符重载全局函数 引用传入后置递增后的myInt,注意易错点:为什么使用引用?

使用引用的原因:解决浅拷贝问题!

如果不加&符号operator<<(ostream& out, MyInteger myint)传入的是对myInt的拷贝,在这个左移运算符重载全局函数运行完输出之后会对这个拷贝对象进行释放,从而运行了这个拷贝对象中的析构函数,提前释放了堆区数据。当test02()运行完成后会对myInt进行释放,从而会再一次运行析构函数去释放堆区的数据从而报错。

注意2:后置递增因为一直是在对temp进行增加,因此无法使用(myint++)++,返回的temp的值,再被<<输入时,只能是值的状态。因为执行完++时temp已经被释放没有内存空间,也就不能产生同地址的引用。

注意3:就算是正常的(a++)++这样的语句也会报错。

后置++操作正常是先引用后递增,所以这里用了一个temp来记录递增之前的值,而不是直接返回原来的数的引用,但这里确实不可以进行链式操作了,因为返回回来的对象不是原来的对象,返回的对象是temp。

总结: 前置递增返回引用,后置递增返回

目录
相关文章
|
3月前
|
编译器 C++
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
C++进阶之路:何为运算符重载、赋值运算符重载与前后置++重载(类与对象_中篇)
36 1
|
4月前
|
程序员 编译器 C++
C++中的运算符重载(Operator Overloading)
C++中的运算符重载(Operator Overloading)
39 1
|
8天前
|
C++
C++(十五) 运算符重载
C++中的运算符重载允许对已有运算符的功能进行重新定义,从而扩展语言功能、简化代码并提升效率。重载遵循特定语法,如 `friend 类名 operator 运算符(参数)`。重载时需注意不可新增或改变运算符数量、语义、优先级、结合性和返回类型。常见示例包括双目运算符 `+=` 和单目运算符 `-` 及 `++`。输入输出流运算符 `&lt;&lt;` 和 `&gt;&gt;` 也可重载。部分运算符只能作为成员函数重载。
|
3月前
|
存储 编译器 C++
【C++】:拷贝构造函数和赋值运算符重载
【C++】:拷贝构造函数和赋值运算符重载
22 1
|
3月前
|
C++ 索引
C++核心技术要点《运算符重载》
C++核心技术要点《运算符重载》
45 2
|
2月前
|
自然语言处理 程序员 C++
C++基础知识(五:运算符重载)
运算符重载是C++中的一项强大特性,它允许程序员为自定义类型(如类或结构体)重新定义标准运算符的行为,使得这些运算符能够适用于自定义类型的操作。这样做可以增强代码的可读性和表达力,使得代码更接近自然语言,同时保持了面向对象编程的封装性。
|
2月前
|
Java 程序员 C++
|
2月前
|
编译器 C++
【C++】详解运算符重载,赋值运算符重载,++运算符重载
【C++】详解运算符重载,赋值运算符重载,++运算符重载
|
3月前
|
编译器 C++
【C++】类和对象③(类的默认成员函数:赋值运算符重载)
在C++中,运算符重载允许为用户定义的类型扩展运算符功能,但不能创建新运算符如`operator@`。重载的运算符必须至少有一个类类型参数,且不能改变内置类型运算符的含义。`.*::sizeof?`不可重载。赋值运算符`=`通常作为成员函数重载,确保封装性,如`Date`类的`operator==`。赋值运算符应返回引用并检查自我赋值。当未显式重载时,编译器提供默认实现,但这可能不足以处理资源管理。拷贝构造和赋值运算符在对象复制中有不同用途,需根据类需求定制实现。正确实现它们对避免数据错误和内存问题至关重要。接下来将探讨更多操作符重载和默认成员函数。
|
3月前
|
C++
c++进阶篇(一)——运算符重载
c++进阶篇(一)——运算符重载