C++类和对象——日期类的实现

时间:2022-08-01 01:09:59

今天我们实现一个日期类,对前面所学知识进行小的应用总结。

首先我们先写日期类Date最基本的东西,私有成员,构造函数,拷贝构造函数,析构函数(可以不写,编译器默认生成),赋值运算符重载。

私有成员变量:
private:
    int _year=1900;
    int _month=1;
    int _day=1;

构造函数:
    Date(int year,int month,int day)
        :_year(year)
        ,_month(month)
        ,_day(day)
    { }
拷贝构造函数:
Date(const Date& d)
{
    _year=d._year;
    _month=d._month;
    _day=d._day;
} 
赋值运算符重载:检测是否自己给自己赋值
// d2 = d3 -> d2.operator=(&d2, d3)
Date& Date::operator=(const Date& d)
{
	if (this != &d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	return *this;
}

我们还要考虑,我们创建的日期是否合法,比如每个月的天数是否符合月数。那我们就要写一个检查日期合法的函数,检查日期又要获得当年当月的天数,又要再写一个获得月份天数的函数。

获得当年月份天数的函数:

	// 获取某年某月的天数
	//频繁调用,放类中定义作inline
	int GetMonthDay(int year, int month)
	{
		static int days[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };//使用static,静态区,在程序结束前只用开辟一次空间
		if (month == 2 &&
			(year % 400 == 0) || (year % 4 == 0 && year % 100 != 0))
		{
			return 29;
		}
		else
		{
			return days[month];
		}
	}

 检查日期:

	bool CheckDate()
	{
		if (_year >= 1 
			&& _month > 0 && _month < 13 
			&& _day>0 && _day <= GetMonthDay(_year, _month))
		{
			return true;
		}
		else
		{
			return false;
		}
	}

Date.h头文件

#pragma once
#include<iostream>
#include<assert.h>
using namespace std;
class Date
{
public:
	// 获取某年某月的天数
	//频繁调用,放类中定义作inline
	int GetMonthDay(int year, int month)
	{
		static int days[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31 };//使用static,静态区,在程序结束前只用开辟一次空间
		if (month == 2 &&
			(year % 400 == 0) || (year % 4 == 0 && year % 100 != 0))
		{
			return 29;
		}
		else
		{
			return days[month];
		}
	}
	bool CheckDate()
	{
		if (_year >= 1 
			&& _month > 0 && _month < 13 
			&& _day>0 && _day <= GetMonthDay(_year, _month))
		{
			return true;
		}
		else
		{
			return false;
		}
	}
	// 全缺省的构造函数
	Date(int year , int month , int day)
		:_year(year)
		,_month(month)
		,_day(day)
	{

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

	// 赋值运算符重载
	// d2 = d3 -> d2.operator=(&d2, d3)
	Date& operator=(const Date& d);


	// 日期+=天数
	Date& operator+=(int day);
	// 日期+天数
	Date operator+(int day) const;
	// 日期-天数
	Date operator-(int day) const;
	// 日期-=天数
	Date& operator-=(int day);

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

	// >运算符重载
	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;

	// 日期-日期 返回天数
	int operator-(const Date& d) const;

private:
	int _year=1900;
	int _month=1;
	int _day=1;
};

下面我们来写Date.cpp文件,定义各运算符重载 

任何一个类只需要写一个> ==或< ==重载,剩下比较运算符重载复用即可。

// >运算符重载
bool Date::operator>(const Date& d) const
{
	if ((_year > d._year)
		|| (_year == d._year && _month > d._month)
		|| (_year == d._year && _month == d._month && _day > d._day))
	{
		return true;
	}
	else
	{
		return false;
	}
}
// ==运算符重载
bool Date::operator==(const Date& d) const
{
	return _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);
}

// !=运算符重载
bool Date::operator != (const Date& d) const
{
	return !(*this == d);
}

实现日期+=天数,日期+天数,日期-=天数,日期-天数。

同样的我们可以,写其中一种另一种进行复用。

这里我们采用写日期+=天数,日期-=天数。

// 日期+=天数 d2+=d1+=100
//自身要改变,并且可以连续操作使用引用返回。
Date& Date::operator+=(int day)
{
	if (_day < 0)
	{
		return *this -= -day;
	}
	_day += day;
	while (_day > GetMonthDay(_year, _month))
	{
		_day -= GetMonthDay(_year, _month);
		++_month;
		if (_month == 13)
		{
			++_year;
			_month = 1;
		}
	}

	return *this;
}

// 日期+天数 d2+100
Date Date::operator+(int day) const
{
	Date ret = *this;
	ret += day;
	return ret;
}

// 日期-天数
Date Date::operator-(int day) const
{
	Date ret(*this);
	ret -= day;
	return ret;
}

// 日期-=天数
Date& Date::operator-=(int day)
{
	if (_day < 0)
	{
		return *this += -day;
	}
	_day -= day;
	while (_day <= 0)
	{
		--_month;
		if (_month == 0)
		{
			_month = 12;
			--_year;
		}
		_day += GetMonthDay(_year, _month);

	}
	return *this;
}

 【问题】

1.这里为什么采用写+=和-=,而不是写+和-?

答:+=重载和-=重载中,并没有使用拷贝构造函数,如果使用+和-,则会使用拷贝构造函数。

Date Date::operator+(int day)
{
	Date ret = *this;
	// ...
	ret._day += day;
	while (ret._day > GetMonthDay(ret._year, ret._month))
	{
		//...
	}

	return ret;
}

// d1 += 100
Date& Date::operator+=(int day)
{
	*this = *this + day;

	return *this;
}

写+的话,内部创建ret拷贝构造,值返回拷贝构造,2次拷贝构造。+=复用中,需要调用+,2次拷贝构造。

写+=的话,内部没有拷贝构造,复用+=,2个拷贝构造,所以综上还是写+=调用拷贝构造的次数少,效率好。

2.日期+天数和日期-天数,为什么是const成员函数?

答:因为日期+天数和日期-天数都不需要改变原来的日期,即this指针指向的内容,所以我们用const修饰,const Date* const this ,使指向内容不变。

 前置++,后置++,前置--,后置--

// 前置++:返回+1之后的结果
 // 注意:this指向的对象函数结束后不会销毁,故以引用方式返回提高效率
Date& Date::operator++()
{
	return *this += 1;
}

// 后置++
 // 前置++和后置++都是一元运算符,为了让前置++与后置++形成能正确重载
 // C++规定:后置++重载时多增加一个int类型的参数,但调用函数时该参数不用传递,编译器自动传
递
 // 注意:后置++是先使用后+1,因此需要返回+1之前的旧值,故需在实现时需要先将this保存一份,
然后给this+1
 // 而temp是临时对象,因此只能以值的方式返回,不能返回引用

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

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

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

 日期-日期 返回天数

// 日期-日期 返回天数
int Date::operator-(const Date& d) const
{
	//思路,用小的一天天加上去
	//假定前一个是大日期,后一个小日期
	Date max = *this;
	Date min = d;
	int flag = 1;
	if (*this < d)
	{
		max = d;
		min = *this;
		flag = -1;
	}
	int n = 0;
	while (min != max)
	{
		++n;
		++min;
	}
	return n * flag;
}

Date.cpp文件内容

#include"Date.h"

// >运算符重载
bool Date::operator>(const Date& d) const
{
	if ((_year > d._year)
		|| (_year == d._year && _month > d._month)
		|| (_year == d._year && _month == d._month && _day > d._day))
	{
		return true;
	}
	else
	{
		return false;
	}
}

// ==运算符重载
bool Date::operator==(const Date& d) const
{
	return _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);
}

// !=运算符重载
bool Date::operator != (const Date& d) const
{
	return !(*this == d);
}

// 赋值运算符重载
// d2 = d3 -> d2.operator=(&d2, d3)
Date& Date::operator=(const Date& d)
{
	if (this != &d)
	{
		_year = d._year;
		_month = d._month;
		_day = d._day;
	}
	return *this;
}

// 日期+=天数 d2+=d1+=100
Date& Date::operator+=(int day)
{
	if (_day < 0)
	{
		return *this -= -day;
	}
	_day += day;
	while (_day > GetMonthDay(_year, _month))
	{
		_day -= GetMonthDay(_year, _month);
		++_month;
		if (_month == 13)
		{
			++_year;
			_month = 1;
		}
	}

	return *this;
}

// 日期+天数 d2+100
Date Date::operator+(int day) const
{
	Date ret = *this;
	ret += day;
	return ret;
}

// 日期-天数
Date Date::operator-(int day) const
{
	Date ret(*this);
	ret -= day;
	return ret;
}

// 日期-=天数
Date& Date::operator-=(int day)
{
	if (_day < 0)
	{
		return *this += -day;
	}
	_day -= day;
	while (_day <= 0)
	{
		--_month;
		if (_month == 0)
		{
			_month = 12;
			--_year;
		}
		_day += GetMonthDay(_year, _month);

	}
	return *this;
}

// 前置++
// 前置++:返回+1之后的结果
 // 注意:this指向的对象函数结束后不会销毁,故以引用方式返回提高效率
Date& Date::operator++()
{
	return *this += 1;
}

// 后置++
 // 前置++和后置++都是一元运算符,为了让前置++与后置++形成能正确重载
 // C++规定:后置++重载时多增加一个int类型的参数,但调用函数时该参数不用传递,编译器自动传
递
// 注意:后置++是先使用后+1,因此需要返回+1之前的旧值,故需在实现时需要先将this保存一份,
然后给this + 1
// 而temp是临时对象,因此只能以值的方式返回,不能返回引用
Date Date::operator++(int)
{
	Date tmp(*this);
	*this += 1;
	return tmp;
}

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

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

// 日期-日期 返回天数
int Date::operator-(const Date& d) const
{
	//思路,用小的一天天加上去
	//假定前一个是大日期,后一个小日期
	Date max = *this;
	Date min = d;
	int flag = 1;
	if (*this < d)
	{
		max = d;
		min = *this;
		flag = -1;
	}
	int n = 0;
	while (min != max)
	{
		++n;
		++min;
	}
	return n * flag;
}

以上就是整个日期类的实现,我们通过实现日期类可以对前面所学默认成员函数进行巩固加强。