| 目录 1.日期类的定义:Date.h 2.成员函数的实现:Date.cpp ?2.1准确获取某年某月有多少天 2.2日期类构造函数 (补充)2.3日期类拷贝构造 2.4赋值运算符重载 2.5+=运算符重载 2.6+运算符重载 2.7-=运算符重载 ?2.8-运算符重载 2.9前置++运算符重载 (补充)2.10前置--运算符重载 (补充)2.11后置-- 运算符重载 2.12后置++运算符重载 2.13>运算符重载 2.14==运算符重载 2.15>=运算符重载 2.16<运算符重载 2.17<=运算符重载 2.18!=运算符重载 2.19.计算两个日期之间的间隔天数,日期减去日期 ? 3.测试:Test.h 
 1.日期类的定义:Date.h#pragma once
#include <iostream>
#include <assert.h>
using std::cout;
using std::cin;
using std::endl;
class Date
{
public:
	//获取某年某月的天数
	int GetMonthDay(int year, int month);
	//全缺省的构造函数
	Date(int year = 0, int month = 1, int day = 0);
	//打印
	void Print();
	//析构、拷贝构造,赋值重载可以不写,默认生成的就够用,像Stack才需要自己写
	//d+100
	//赋值运算符重载
	Date& operator+=(int day);
	Date operator+(int day);
	Date& operator-=(int day);
	Date operator-(int day);
	//返回两个日期之间相隔的具体天数
	int operator-(const Date& d);
	//++d -> d.operator++(&d)
	Date& operator++();
	//d++ -> d.operator++(&d,0)
	//int参数不需要给实参,因为没用,它的作用是为了和前置++构成函数重载
	Date operator++(int);
 
	//运算符重载
	bool operator>(const Date& d);
	bool operator<(const Date& d);
	bool operator>=(const Date& d);
	bool operator<=(const Date& d);
	bool operator==(const Date& d);
	bool operator!=(const Date& d);
private:
	int _year;
	int _month;
	int _day;
};
 2.成员函数的实现:Date.cpp?2.1准确获取某年某月有多少天inline int GetMonthDay(int year, int month)
{
	
	//数组存储平年每个月的天数
	static int dayArray[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
	int i = dayArray[month];
	if (month == 2 && ((year%4==0&&year%100!=0)||(year%400 == 0)))
	{
		i = 29;
	}
	return i;
}
 2.2日期类构造函数
Date::Date(int year, int month, int day)//缺省参数在声明和实现两边只有一边可用
{
	//检查日期的合法性
	if (year>=0 &&month>0&&month<13 &&day>0&&day<=GetMonthDay(year,month))
	{
		_year = year;
		_month = month;
		_day = day; 
	}
	else
	{
		//严格来说抛异常更好
		cout << "非法日期" << endl;
		cout << year << "年" << month << "月" << day << "日" << endl;
	}
}
 (补充)2.3日期类拷贝构造
Date::Date(const Date& d)
{
    _year = d._year;
    _month = d._month;
    _day = d._day;
}
 2.4赋值运算符重载
Date& Date:: operator=(const Date& d)
{
    _year = d._year;
    _month = d._month;
    _day = d._day;
    return *this;   //这个可不能少,少了遇上连续赋值语句a=b=c会出错。
}
 2.5+=运算符重载
 
 考虑越界情况 如果_day加了整数以后,<=该月最大天数,则不需要修改,直接返回该日期.如果_day加了整数以后,>该月最大天数,则_day减去新的月所拥有的最大天数,然后该月加1如果执行了第二步后,_day仍大于新的月所拥有天数,继续执行第二步,并且循环
 //类似加法进位的规则:1.天满了,减去当前月的天数,月+1 
//2.月满了,年+1,月置成1
Date& Date::operator+=(int day)
{
	if (day < 0)
	{
		*this -= -day;
	}
	else
	{
		_day += day;
		//天数不合法,不断进位使其合法
		while (_day > GetMonthDay(_year, _month))
		{
			_day -= GetMonthDay(_year, _month);
			_month++;
			if (_month > 12)
			{
				++_year;
				_month = 1;
			}
		}
	}
	 
	return *this;
 
}
 2.6+运算符重载Date Date::operator+(int day)
{
	Date ret(*this);
	//复用operator+=
	ret += day; //相当于ret.operator+=(day)
	return ret;
}
 2.7-=运算符重载
Date& Date::operator-=(int day)
{
	if(day<0)
	{
		//_day += -day;
		天数不合法,不断进位使其合法
		//while (_day > GetMonthDay(_year, _month))
		//{
		//	_day -= GetMonthDay(_year, _month);
		//	_month++;
		//	if (_month > 12)
		//	{
		//		++_year;
		//		_month = 1;
		//	}
		//}
		//复用调用上面的函数
		* this += -day;
	}
	else
	{
		/*_day -= day;
		while (_day <= 0)
		{
			--_month;
			if (_month == 0)
			{
				--_year;
				_month = 12;
			}
			_day += GetMonthDay(_year, _month);
		}*/
		* this -= day;
	}
	 
	return *this;
}
 ?
 2.8-运算符重载
Date Date::operator-(int day)
{
	Date tmp = *this;
	tmp -= day;//tmp.operator
	return tmp;
}
 2.9前置++运算符重载
//++d -> d.operator++(&d)
Date& Date::operator++()
{
	*this += 1;
	return *this;
}
 (补充)2.10前置--运算符重载
Date& Date::operator--()
{
    *this -= 1;
    return *this;
}
 (补充)2.11后置-- 运算符重载
Date& Date::operator--(int)  //这个int用于区分前置还是后置
{
    Date tmp(*this);
    *this -= 1;
    return tmp;
}
 2.12后置++运算符重载
//d++ -> d.operator++(&d,0)
//int参数不需要给实参,因为没用,它的作用是为了和前置++构成函数重载
Date Date::operator++(int)
{
    Date tmp(*this);
    *this += 1;
    return tmp;
}
 2.13>运算符重载
bool Date::operator>(const Date& d)
{
	if (_year > d._year)
	{
		return true;
	}
	else if(_year == d._year)
	{
		if (_month > d._month)
		{
			return true;
		}
		else if (_month == d._month)
		{
			if (_day > d._day)
			{
				return true;
			}
		}
	}
	return false;
}
 2.14==运算符重载
bool Date::operator==(const Date& d)
{
	return _year = d._year
		&& _month == d._month
		&& _day == d._day;
}
 2.15>=运算符重载
bool Date::operator>=(const Date& d)
{
	return *this > d || *this == d;
}
 2.16<运算符重载
bool Date::operator<(const Date& d)
{
	return !(*this >= d);
}
 2.17<=运算符重载
bool Date::operator<(const Date& d)
{
	return !(*this >= d);
}
 2.18!=运算符重载
bool Date::operator!=(const Date& d)
{
	return !(*this == d);
}
 2.19.计算两个日期之间的间隔天数,日期减去日期
//调用前面实现的各种函数来实现
int Date::operator-(const Date& d)
{
	//效率差别不大的情况下,尽量选择可读性强的,简单的程序
	Date max = *this;
	Date min = d;
	int flag = 1;
	if (*this<d)
	{
		max = d;
		min = *this;
		flag = -1;
	}
	int n = 0;
	while (min != max)
	{
		++min;
		++n;
	}
	return n * flag;
}
 ?
3.测试:Test.h#include "Date.h"
void Test1()
{
	Date d1(2021, 5, 25);
	d1.Print();
	Date d2(2021, 0, 0);
	//d2.Print();
	Date d3(2021, 2, 29);
	d3.Print();
}
void Test2()
{
	Date d1(2021, 5, 25);
	d1.Print();
	d1 += 3;
	d1.Print();
	d1 += 17;
	d1.Print();
}
void Test3()
{
	Date d1(2021, 5, 27);
	d1 -= 120;
	d1.Print();
	Date d2(2021, 5, 27);
	d2 -= -100;
	d2.Print();
	Date d3(2021, 5, 27);
	d3 += 100;
	d3.Print();
	Date d4(2021, 5, 27);
	d4 += -100;
	d4.Print();
}
void Test4()
{
	Date d1(2021, 5, 27);
	//前置++和后置++都完成了++,不同的地方在于返回值不一样
	//因为他们的运算符是一样的,函数名就是一样的,
	//为了区分,对后置++做了特殊处理,加了一个参数,形成函数重载
	Date ret1 = d1++;//d1.operator++(&d1)
	ret1.Print();
	d1.Print();
	Date ret2 = ++d1;//d1.operator++(&d1,0)
	ret2.Print();
	d1.Print();
}
void Test5()
{
	Date d1(2021, 5, 27);
	Date d2(2021, 12, 31);
	cout << d2 - d1 << endl;
	cout << d1 - d2 << endl;
}
int main()
{
	 
	Test5();
	return 0;
}
 |