C++日期类(Date类)的实现以及相关运算符重载

日期类(Date类)

先看.h文件

Date.h

#pragma once
#include 
using namespace std;

class Date
{
public:
	int GetMonthDay(int year, int month) const;
	// 四个成员函数
	Date(int year, int month, int day);

	bool operator==(const Date& d) const;
	bool operator!=(const Date& d) const;

	bool operator>(const Date& d) const;
	bool operator<(const Date& d) const;
	bool operator>=(const Date& d) const;
	bool operator<=(const Date& d) const;

	Date& operator=(const Date& d);

	//d1 + 10
	Date operator+(int day) const;
	Date operator+=(int day);

	Date operator-(int day) const;
	Date operator-=(int day);

	// 前置
	Date& operator++();
	Date& operator--();

	// 后置
	Date operator++(int);
	Date operator--(int);

	// d1-d2
	int operator-(const Date& d) const;

	void Print() const;

private:
	int _year;
	int _month;
	int _day;
};

void Test1();
void Test2();
void Test3();
void Test4();

非复用.cpp

#include "Date.h"

int Date::GetMonthDay(int year, int month) const
{
	int monthArray[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };  //存储每月的天数,2月暂时用28天表示
	if (month == 2 && ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0))   //判断是否闰年,闰年的话2月变成29天
		return 29;
	else
		return monthArray[month];
}

Date::Date(int year = 2019, int month = 1, int day = 1)      //构造函数
{
	if (year > 0 &&											//判断日期是否合法
		month > 0 && month < 13 &&
		day > 0 && day <= GetMonthDay(year, month))
	{
		_year = year;
		_month = month;
		_day = day;
	}
	else
	{
		cout << "日期非法" << endl;
	}

}

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

bool Date::operator!=(const Date& d) const
{
	return this->_day != d._day ||
		this->_month != d._month ||
		this->_year != d._year;
}

bool Date::operator>(const Date& d) const
{
	return (this->_year > d._year || (this->_year == d._year&&this->_month > d._month) ||
		(this->_year == d._year&&this->_month == d._month&&this->_day > d._day));
}

bool Date::operator<(const Date& d) const
{
	return (this->_year < d._year || (this->_year == d._year&&this->_month < d._month) ||
		(this->_year == d._year&&this->_month == d._month&&this->_day < d._day));
}

bool Date::operator>=(const Date& d) const
{
	return *this < d ? 0 : 1;
}

bool Date::operator<=(const Date& d) const
{
	return *this > d ? 0 : 1;
}

Date& Date::operator=(const Date& d)
{
	if(*this != d)
	{
		this->_year = d._year;
		this->_month = d._month;
		this->_day = d._day;
	}
		
	return *this;
}

//d1 + 10
Date Date::operator+(int day) const
{
	if (day < 0)                         //用户若输入负数,则表示向前推day天,直接调用"-"
		return *this - (-day);
	Date ret(*this);
	while (ret._day + day > GetMonthDay(ret._year, ret._month))  //day超过本月剩余天数,则进入循环
	{
		day = day - (GetMonthDay(ret._year, ret._month) - ret._day);
		ret._day = 0;
		if (ret._month == 12)  //判断是否到下一年
		{
			ret._year++;
			ret._month = 0;
		}
		ret._month++;  //进入下一个月
	}
	ret._day += day;  //加上剩余的天数

	return ret;
}

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

	return *this;
}

Date Date::operator-(int day) const
{
	if (day < 0)							//若用户输入负数,则表示向后推day天,直接调用"-"
		return *this + (-day);
	Date ret(*this);
	while (ret._day < day)                  //当本月天数不够减day时,进入循环
	{
		day = day - ret._day;
		if (ret._month == 1)                //判断是否退回前一年
		{
			ret._month = 13;
			ret._year--;
		}
		ret._month--;                       //向前退一个月
		ret._day = GetMonthDay(ret._year, ret._month);
	}
	ret._day -= day;

	return ret;
}

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

	return *this;
}

// 前置
Date& Date::operator++()
{
	if (++_day > GetMonthDay(_year, _month))
	{
		_month++;
		if (_month == 13)
		{
			_month = 1;
			_year++;
		}

		_day = 1;
	}

	return *this;
}

Date& Date::operator--()
{
	if (--_day == 0)
	{
		_month--;
		if (_month == 0)
		{
			_month = 12;
			_year--;
		}

		_day = GetMonthDay(_year, _month);
	}
			
	return *this;
}

// 后置
Date Date::operator++(int)
{
	Date ret(*this);
	if (++_day > GetMonthDay(_year, _month))
	{
		_month++;
		if (_month == 13)
		{
			_month = 1;
			_year++;
		}

		_day = 1;
	}

	return ret;
}

Date Date::operator--(int)
{
	Date ret(*this);
	if (--_day == 0)
	{
		_month--;
		if (_month == 0)
		{
			_month = 12;
			_year--;
		}

		_day = GetMonthDay(_year, _month);
	}

	return ret;
}

// d1-d2
int Date::operator-(const Date& d) const
{
	int bYear = _year;
	int sYear = d._year;
	int bMonth = _month;
	int sMonth = d._month;
	int bDay = _day;
	int sDay = d._day;
	int yFlag, mFlag, dFlag;
	yFlag = mFlag = dFlag = 1;

	//本年x月最后一天之前有几天
	int DaysBeforeMonth[] = { 0,31,59,90,120,151,181,212,243,273,304,334,365 };

	if (_year < d._year)
	{
		bYear = d._year;
		sYear = _year;
		yFlag = -1;
	}

	if (_month < d._month)
	{
		bMonth = d._month;
		sMonth = _month;
		mFlag = -1;
	}

	if (_day < d._day)
	{
		bDay = d._day;
		sDay = _day;
		dFlag = -1;
	}

	int years = 0;
	int days = 0;//闰年数

	years = bYear - sYear;

	for (int i = sYear; i <= bYear; i++)
	{
		if ((i % 4 == 0 && i % 100 != 0) || i % 400 == 0)
		{
			days++;
		}
	}
	
	return (bDay - sDay)*dFlag +
		(DaysBeforeMonth[bMonth] - DaysBeforeMonth[sMonth])*mFlag +
		(years * 365 + days)*yFlag;
}

void Date::Print() const
{
	cout << _year << "-" << _month << "-" << _day << endl;
}


// == / != / >= / <=
void Test1()
{
	Date d1(2019, 1, 15);
	d1.Print();
	Date d2(2021, 10, 10);
	d2.Print();
	Date d3(2021, 10, 10);
	d3.Print();
	cout << (d1 == d2) << endl;
	cout << (d2 == d3) << endl;
	cout << (d1 != d3) << endl;
	cout << (d2 != d3) << endl;

	cout << (d1 >= d2) << endl;
	cout << (d2 >= d3) << endl;
	cout << (d2 <= d3) << endl;
	cout << (d1 <= d2) << endl;
}

// + / - / += / -=
void Test2()
{
	Date d1(2019, 1, 15);
	d1.Print();
	Date d2(2021, 10, 10);
	d2.Print();
	Date d3 = (d1 + 999);
	d3.Print();
	Date d4 = (d2 - 999);
	d4.Print();

	d1 += 999;
	d2 -= 999;
	d1.Print();
	d2.Print();
}

// ++ / -- (前置/后置)
void Test3()
{
	Date d1(2018, 5, 10);
	Date d2(2019, 2, 5);
	d1--.Print();
	d1.Print();
	d2++.Print();
	d2.Print();

	(--d1).Print();
	d1.Print();
	(++d2).Print();
	d2.Print();
}

void Test4()
{
	Date d1(2019, 1, 15);
	Date d2(2021, 10, 10);
	int a = (d2 - d1);
	cout << a << endl;
}


int main()
{
	//Test1();
	//Test2();
	//Test3();
	Test4();

	system("pause");
	return 0;
}

其实在经过分析.h文件后,会发现有大量的函数代码可以复用实现,只需要实现"=="、">"、"+="、"-="几个函数就可以轻易实现其它函数。

复用.cpp

#include "Date.h"

int Date::GetMonthDay(int year, int month) const
{
	int monthArray[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))
		return 29;
	else
		return monthArray[month];
}

Date::Date(int year = 2019, int month = 1, int day = 1)
{
	if (year > 0 &&
		month > 0 && month < 13 &&
		day > 0 && day <= GetMonthDay(year, month))
	{
		_year = year;
		_month = month;
		_day = day;
	}
	else
	{
		cout << "日期非法" << endl;
	}

}

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

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

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

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

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

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

Date& Date::operator=(const Date& d)
{
	if (*this != d)
	{
		this->_year = d._year;
		this->_month = d._month;
		this->_day = d._day;
	}

	return *this;
}

//d1 + 10
Date Date::operator+(int day) const
{
	Date ret(*this);
	ret += day;

	return ret;
}

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

	return *this;
}

Date Date::operator-(int day) const
{
	Date ret(*this);
	ret -= day;

	return ret;
}

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

	return *this;
}

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

	return *this;
}

Date& Date::operator--()
{
	*this -= 1;

	return *this;
}

// 后置
Date Date::operator++(int)
{
	Date ret(*this);
	*this += 1;

	return ret;
}

Date Date::operator--(int)
{
	Date ret(*this);
	*this -= 1;

	return ret;
}

// d1-d2
int Date::operator-(const Date& d) const
{
	Date max(*this);
	Date min(d);
	int flag = 1;
	int days = 0;//相差天数

	if (*this < d)
	{
		max = d;
		min = *this;
		flag = -1;
	}

	while (max != min)
	{
		max--;
		days++;
	}

	return days;
}

void Date::Print() const
{
	cout << _year << "-" << _month << "-" << _day << endl;
}



// == / != / >= / <=
void Test1()
{
	Date d1(2019, 1, 15);
	d1.Print();
	Date d2(2021, 10, 10);
	d2.Print();
	Date d3(2021, 10, 10);
	d3.Print();
	cout << (d1 == d2) << endl;
	cout << (d2 == d3) << endl;
	cout << (d1 != d3) << endl;
	cout << (d2 != d3) << endl;

	cout << (d1 >= d2) << endl;
	cout << (d2 >= d3) << endl;
	cout << (d2 <= d3) << endl;
	cout << (d1 <= d2) << endl;
}

// + / - / += / -=
void Test2()
{
	Date d1(2019, 1, 15);
	d1.Print();
	Date d2(2021, 10, 10);
	d2.Print();
	Date d3 = (d1 + 999);
	d3.Print();
	Date d4 = (d2 - 999);
	d4.Print();

	d1 += 999;
	d2 -= 999;
	d1.Print();
	d2.Print();
}

// ++ / -- (前置/后置)
void Test3()
{
	Date d1(2018, 5, 10);
	Date d2(2019, 2, 5);
	d1--.Print();
	d1.Print();
	d2++.Print();
	d2.Print();

	(--d1).Print();
	d1.Print();
	(++d2).Print();
	d2.Print();
}

void Test4()
{
	Date d1(2019, 1, 15);
	Date d2(2021, 10, 10);
	int a = (d2 - d1);
	cout << a << endl;
}


int main()
{
	//Test1();
	//Test2();
	//Test3();
	Test4();

	system("pause");
	return 0;
}

由对比发现,复用代码比非复用代码断了许多,说明了在写代码之前要先对大局进行分析,避免不必要的代码冗余。

你可能感兴趣的:(C++日期类(Date类)的实现以及相关运算符重载)