C++运算符重载

目录

前言

一、有关运算符重载的知识回顾

1.并不是所有运算符都可以重载

 2.一部分运算符必须定义为成员函数

3.运算符如何实现重载

二、实现一个简单的日期类并介绍一些特殊情况

1.日期类的实现

2.特殊情况——前置++与后置++如何区分

3.日期类代码一览


前言

        本文是作为C++类与对象文章的补充,后续还会补充其他内容。       ——2024.2.17


一、有关运算符重载的知识回顾

1.并不是所有运算符都可以重载

        除了图1所示的运算符外,一些在C语言中没有出现的运算符也不可以进行重载比如“@”、“ 。”等(注意:中文字符都不可以重载) 

 2.一部分运算符必须定义为成员函数

3.运算符如何实现重载

        借助关键字operator进行重载,格式如图3所示:[返回类型] operator[重载运算符](参数)

二、实现一个简单的日期类并介绍一些特殊情况

1.日期类的实现

         在开始介绍之前,要先解释一下本文的日期类是什么意思,日期类就是字面意思一个描述日期的类,什么是类来着??(如果你忘记了,可以看看博主的这篇文章click_me),类是一种由用户定义的自定义类型,对于一个类而言其不仅仅有成员变量还有成员函数,而类的精华就在于其成员函数(成员变量也很重要!!!不过本文更想讲解一下有关日期类的成员函数),那么对于一个日期来说加减乘除哪一个是有意义的呢??

        显然,在生活中很少见到日期和日期作“乘除法”,但是日期的“加”和“减”,我们是常常见到的。比如,在你规划接下来的哪天陪女朋友度过时,你会计算出那一天和今天相差几天。这就涉及到了日期的加减法。那么接下来我们试着构造一个日期类:

        首先我们需要几个描述日期的成员变量:_year、_month、_day分别表示日期的年、月、日:

        然后呢,我们考虑类的默认成员函数,首先我们需要考虑一下这个类的构造函数,对于日期来说,年份我们默认大于0;月份大于0,小于13 ;每月日期是不固定的大月31天、小月30天、二月份(闰年)29或28天(平年)。显然,每月日期的确定较为复杂。不过我们可以试着使用一个函数确定一个月有多少天:

         解释一下图5代码,首先定义一个大小为13的整形数组,用来下标代表月份,其中下标0,表示0月,现实中没有0月,把0月设置成0天,而后对于任何一年,除了二月份其他月份有多少天是确定,所以在对应月份的下标所指向的空间上存储相应的天数,比如下标1代表一月份,所以_arrt[1]=31,也就是一月份有31天,以此类推,将其他月份天数填入数组即可。 

        接下来,我们就可以来实现一下日期类的构造函数:

        由于没有向系统申请过一些诸如“堆空间”等资源的情况,所以我们简单写一个析构函数,并在其中打印信息,起到提示我们调用过析构函数。

         拷贝构造函数也是如此,由于没有申请需要释放的资源(特指“堆空间”等资源),所以我们无需考虑,会对同一内存空间析构两次的风险(如果对我的说法感到疑惑,请看博主的这篇文章的第三部分的第3点——拷贝构造函数click_me),所以完成对数据的值拷贝即可:

        那么还有最后赋值拷贝函数,这个函数就涉及到了运算符重载,本文就由此开始对运算符重载的讲解:

        赋值运算符重载也无需考虑会对同一内存释放两次的问题,所以完成值拷贝即可:

         ==重载(等于):这个实现比较简单,当年、月、日都相等的时候两个日期相等

        >重载(大于): 当先判断两个日期的年分,年份大的就大,如果相等,则比月份,月份大就大,对于天也是如此。

        <重载(小于):当实现了以上两个运算符后我们可以“偷个懒”,当一个日期既不等于也不大于一个日期时,是不是就是小于呢

        <=重载(小于等于):当一个日期不大于另一个日期时,是不是就满足条件了呢

        >=重载(大于等于):当一个日期不小于另一个日期时。是不是就满足了条件了呢

        +=重载(加等):对于加等运算符重载(本文实现的是在日期的“天”作加),我们可以考虑,当+后日期没有超过本月的最大合法天数,且不小于1,则此时天合理,如果+后超过本月最大合法天数,则需要在月分上进1,若月份满12,在+后需要对年进1。此外我们还要考虑,加的内容是不一定为正的,也就是说也可以加的是负数,这一处,我们暂且搁置,我们可以在实现-=后复用-=,此外需要主语,+=运算符重载后需不需要返回值呢?我们可以对标整数的+=,整数+=后的结果是可以赋值给其他变量的。

        -=重载(减等):对于减等运算符重载(本文实现的是在日期的“天”作减),我们可以考虑,当-后日期没有小于本月的最小合法天数,且不大于最大和法天数,则此时天合理,如果-后超过本月最小合法天数,则需要在月分借进1,若月份为1,在-后可能需要对年借1。此外我们还要考虑,减的内容是不一定为正的,也就是说也可以减的是负数,这一处,我们暂且搁置,我们可以在实现+=后复用+=,此外注意-=的返回值。于+=相似。

        -重载(减法):减与减等的区别在于减不会影响到被减数本身,就像5-3=2,5还是5,它并不会因为参与了减法而变为其他值,举个例子:

 而且我们也应该注意到加法也有返回值。减法与减等的区别是,减等会影响到被减数本身,而减法不会,所以我们只需要拷贝一份被减数,令被减数的拷贝复用-=并返回-=后的结果即可。

        +重载(加法):加法的逻辑与减法相似,这里就不过多赘述了,此外无需考虑天数可能越过合理值的问题,因为复用的-=或+=已经考虑并解决了该问题

2.特殊情况——前置++与后置++如何区分

        根据运算符重载的格式([返回类型] operator[重载运算符](参数)),我们似乎很难区分前置++与后置++。编译器给了我们这样一种规定:传入参数是为后置,不传入参数时为前置。

        前置++:Date& operator++();

        后置++:Date& operator++(int);

        前置--与后置--与上述相似:

        

 

3.日期类代码一览

#include <iostream>

using namespace std;
class Date

{

public:

	// 获取某年某月的天数
	int GetMonthDay(int year, int month)
	{
		int arry[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
		if (month == 2 && (year % 4 == 0 && year % 100 != 0 || year % 400 == 0))
		{
			arry[2]++;
		}
		return arry[month];
	}
	// 全缺省的构造函数
	Date(int year = 1, int month = 1, int day = 1)
	{
		if (year < 1)
		{
			cout << "error date" << endl;
			return;
		}
		else if (month > 12 || month < 1)
		{
			cout << "error date" << endl;
			return;
		}
		else if (day<1 || day > GetMonthDay(year, month))
		{
			cout << "error date" << endl;
			return;
		}
		_year = year;
		_month = month;
		_day = day;
		//cout << "Date()" << endl;
	}

	// 拷贝构造函数
	Date(const Date& d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}


	// 赋值运算符重载

  // d2 = d3 -> d2.operator=(&d2, d3)

	Date& operator=(const Date& d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
		return *this;
	}

	// 析构函数
	~Date()
	{
		//cout << "~Date()" << endl;
	}

	// 日期+=天数

	Date& operator+=(int day)
	{
		if (day < 0)
		{
			day = -day;
			*this -= day;
			return *this;
		}
		_day += day;
		while (_day > GetMonthDay(_year, _month))
		{
			_day -= GetMonthDay(_year, _month);
			_month++;
			if (_month > 12)
			{
				_month = 1;
				_year++;
			}
		}
		
		return *this;
	}


	// 日期+天数

	Date operator+(int day)
	{
		Date tmp(*this);
		tmp += day;
		return tmp;
		
	}



	// 日期-天数

	Date operator-(int day)
	{
		Date tmp(*this);
		tmp -= day;
		return tmp;
	}



	// 日期-=天数

	Date& operator-=(int day)
	{
		if (day < 0)
		{
			day = -day;
			*this += day;
			return *this;
		}
		_day -= day;
		while (_day < 0)
		{
			_day += GetMonthDay(_year, --_month);
			if (_month <= 1&&_day<0)
			{
				_year--;
				_month = 13;
			}
			
		}
		return *this;
	}



	// 前置++
	Date& operator++()
	{
		*this += 1;
		return *this;
	}

	// 后置++
	Date operator++(int)
	{
		Date tmp(*this);
		*this += 1;
		return tmp;
	}

	// 后置--

	Date operator--(int)
	{
		Date tmp(*this);
		*this -= 1;
		return tmp;
	}



	// 前置--
	Date& operator--()
	{
		*this -= 1;
		return *this;
	}



	// >运算符重载

	bool operator>(const Date& d)
	{
		if (_year > d._year)
		{
			return true;
		}
		else if (_month > d._month)
		{
			return true;
		}
		else if (_day > d._day)
		{
			return true;
		}

		return false;
	}



	// ==运算符重载

	bool operator==(const Date& d)
	{
		return _year == d._year &&
			_month == d._month &&
			_day == d._day;
	}



	// >=运算符重载

	bool operator >= (const Date& d)
	{
		return !(*this<d);
	}



	// <运算符重载

	bool operator < (const Date& d)
	{
		return !((*this > d) || (*this == d));
	}



	// <=运算符重载

	bool operator <= (const Date& d)
	{
		return !(*this > d);
	}



	// !=运算符重载

	bool operator != (const Date& d)
	{
		return !(*this == d);
	}



	// 日期-日期 返回天数

	int operator-(const Date& d)
	{
		Date Max = *this;
		Date Min = d;
		if (*this < d)
		{
			Max = d;
			Min = *this;
		}
		int count = 0;
		while (Max != Min)
		{
			Min++;
			count++;
		}
		return count;
	}

	void getdate()
	{
		cout << _year << endl;
		cout << _month << endl;
		cout << _day << endl;
	}
private:

	int _year;
	int _month;
	int _day;
};


                                                                                                                                ——本文【完】

  • 20
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

木鱼不是木鱼

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值