C++日期类计算器怎么实现

寻技术 C/C++编程 2023年10月12日 77

这篇文章主要介绍“C++日期类计算器怎么实现”,在日常操作中,相信很多人在C++日期类计算器怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++日期类计算器怎么实现”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!

    1.获取某年某月的天数

    int GetMonthDay(int year, int month)
    {
    	static int monthDayArray[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 monthDayArray[month];
    	}
    }

    2.构造函数

    Date(int year = 1, int month = 1, int day = 1)
    {
    		_year = year;
    		_month = month;
    		_day = day;
    		//检查日期是否合法
    		if (!((year >= 1)
    			&& (month >= 1 && month <= 12)
    			&& (day >= 1 && day <= GetMonthDay(year, month))))
    		{
    			cout << "非法日期" << endl;
    		}
    }

    3.拷贝构造函数

    // 拷贝构造函数 形参加const 防止写反了 问题就可以检查出来了
    Date(const Date& d)
    {
    	_year = d._year;
    	_month = d._month;
    	_day = d._day;
    }

    4.赋值运算符重载

    //d1 = d2 
    //注:1.要注意两个参数的顺序 2.这里面参数不加引用不会导致无穷递归 但为了避免拷贝构造最好加引用
    Date& operator=(const Date& d)
    {
    	//为了支持链式赋值 if是为了避免自己给自己赋值 d1 = d1
    	if (this != &d)
    	{
    		_year = d._year;
    		_month = d._month;
    		_day = d._day;
    	}
    	return *this;
    }

    5.析构函数

    ~Date()//可不写
    {
        ;
    }

    日期类因为没有申请资源,所以无需写析构函数,编译器默认生成的析构函数就可以。

    6.日期+=天数

    //d1 += 100
    //天满了进月 月满了进年 
    Date& operator+=(int day)
    {
    	//避免 d1 += -1000的情形
    	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;
    }

    7.日期+天数

    //d1 + 100
    Date operator+(int day) const
    {
    	Date ret(*this);
    	ret += day;//ret.operator+=(day)
    	return ret;
    }

    8.日期-天数

    //d1 - 100
    Date operator-(int day) const
    {
    	Date ret(*this);
    	ret -= day;
    	return ret;
    }

    9.日期-=天数

    //d1 -= 100
    Date& operator-=(int day)
    {
    	//避免 d1 -= -1000
    	if (day < 0)
    	{
    		return *this += -day;
    	}
    	_day -= day;
    	while (_day <= 0)
    	{
    		--_month;
    		if (_month == 0)
    		{
    			--_year;
    			_month = 12;
    		}
    		_day += GetMonthDay(_year, _month);
    	}
    	return *this;
    }

    10.前置++的运算符重载

    //前置++
    Date& operator++()
    {
    	//会调用 operator+=(int day)
    	*this += 1;
    	return *this;
    }

    11.后置++的运算符重载

    //后置++ ―多一个int参数主要是为了和前置++进行区分 构成函数重载
    Date operator++(int)
    {
    	Date tmp(*this);
    	*this += 1;
    	return tmp;
    }

    12.前置--的运算符重载

    //前置--
    Date& operator--()
    {
    	//复用运算符重载-=
    	*this -= 1;
    	return *this;
    }

    13.后置--的运算符重载

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

    14.>的运算符重载

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

    15.<的运算符重载

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

    16.==的运算符重载

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

    17.>=的运算符重载

    //d1 >= d2
    bool operator>=(const Date& d) const
    {
    	return *this > d || *this == d;
    }

    18.<=的运算符重载

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

    19.!=的运算符重载

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

    20.<<的运算符重载

    //内联函数和静态成员一样 调用处展开 不进符号表
    inline ostream& operator<<(ostream& out, const Date& d)
    {
    	out << d._year << "年" << d._month << "月" << d._day << "日" << endl;
    	return out;
    }

    21.>>的运算符重载

    //cin >> d1 编译器转化成operator(cin,d1) 形参中相比<< 去掉了const
    inline istream& operator>>(istream& in, Date& d)
    {
    	in >> d._year >> d._month >> d._day;
    	return in;
    }

    22.日期-日期

    //日期-日期
    int operator-(const Date& d) const
    {
    	Date max = *this;
    	Date min = d;
    	int flag = 1;
    	if (*this < d)
      //总结:凡是内部不改变成员变量 也就是不改变*this数据的 这些成员函数都应该加const
      //if (d > *this)
    	{
    		max = d;
    		min = *this;
    		flag = -1;
    	}
    	int n = 0;
    	while (min != max)
    	{
    		++n;
    		//复用++ ++到和d1日期相等 就是相差多少天
    		++min;
    	}
    	return n * flag;
    }

    Date.h

    #pragma once
    #include <iostream>
    using namespace std;
    class Date
    {
    	//友元声明(类的任意位置)声明友元时可以不用加inline
    	friend ostream& operator<<(ostream& out, const Date& d);
    	friend istream& operator>>(istream& in, Date& d);
    public:
    	int GetMonthDay(int year, int month)
    	{
    		static int monthDayArray[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 monthDayArray[month];
    		}
    	}
    	Date(int year = 1, int month = 1, int day = 1)
    	{
    		_year = year;
    		_month = month;
    		_day = day;
    		//检查日期是否合法
    		if (!((year >= 1)
    			&& (month >= 1 && month <= 12)
    			&& (day >= 1 && day <= GetMonthDay(year, month))))
    		{
    			cout << "非法日期" << endl;
    		}
    	}
    	// 拷贝构造函数 形参加const 防止写反了 问题就可以检查出来了
    	Date(const Date& d)
    	{
    		_year = d._year;
    		_month = d._month;
    		_day = d._day;
    	}
    	//d1 == d2
    	bool operator==(const Date& d) const;
    	//d1 > d2
    	bool operator>(const Date& d) const;
    	//d1 >= d2
    	bool operator>=(const Date& d) const;
    	//d1 <= d2
    	bool operator<=(const Date& d) const;
    	//d1 < d2
    	bool operator<(const Date& d) const;
    	//d1 != d2
    	bool operator!=(const Date& d) const;
    	//d1 += 100
    	Date& operator+=(int day);
    	//d1 + 100
    	Date operator+(int day) const;
    	//d1 = d2 注:1.要注意两个参数的顺序 2.这里面参数不加引用不会导致无穷递归 但为了避免拷贝构造最好加引用
    	Date& operator=(const Date& d)
    	{
    		//为了支持链式赋值 if是为了避免自己给自己赋值 d1 = d1
    		if (this != &d)
    		{
    			_year = d._year;
    			_month = d._month;
    			_day = d._day;
    		}
    		return *this;
    	}
    	//d1 -= 100
    	Date& operator-=(int day);
    	//d1 - 100
    	Date operator-(int day) const;
    	//++的操作数只有一个 不传参
    	//前置++
    	Date& operator++();
    	//编译器为了区分前置++和后置++ 规定在后置的函数上加了一个参数
    	//后置++
    	Date operator++(int);
    	//允许成员函数加const 此时this指针的类型为:const Date* const this
    	void Print() const
    	{
    		cout << _year << "/" << _month << "/" << _day << endl;
    	}
    	//前置--
    	Date& operator--();
    	//后置--
    	Date operator--(int);
    	//日期-日期
    	int operator-(const Date& d) const;
    	//流插入
    	//d1 << cout编译器会转化成d1.operator<<(cout) this指针抢了左操作数d1的位置
    	//<<和>>的重载一般不写成成员函数 因为this默认抢了第一个参数的位置 Date类对象就是左操作数 不符合使用习惯和可读性
    	/*void operator<<(ostream& out)
    	{
    		out << _year << "年" << _month << "月" << _day << "日" << endl;
    	}*/
    	//取地址重载
    	Date* operator&()
    	{
    		return this;
    	}
    	//const成员取地址重载
    	const Date* operator&() const
    	{
    		return this;
    	}
    	//取地址重载和const成员取地址重载不实现 编译器会默认生成
    private:
    	int _year;
    	int _month;
    	int _day;
    };
    //结论:对于自定义类型,尽量用前置,减少拷贝,提高效率
    //全局函数调用:cout << d1转化成operator<<(cout,d1)
    //全局函数的定义和全局变量不能放在.h文件中 因为函数的定义在Date.cpp和test.cpp都会展开 函数地址进入符号表 链接器链接两个.cpp文件时相同的函数地址会报错
    //解决方法:1.改成静态 2.声明和定义分离
    //static修饰函数只在当前文件可见 不会进入符号表
    //static void operator<<(ostream& out,const Date& d)
    //{
    //	out << d._year << "年" << d._month << "月" << d._day << "日" << endl;
    //}
    //ostream& operator<<(ostream& out, const Date& d);
    //内联函数和静态成员一样 调用处展开 不进符号表
    inline ostream& operator<<(ostream& out, const Date& d)
    {
    	out << d._year << "年" << d._month << "月" << d._day << "日" << endl;
    	return out;
    }
    //cin >> d1 编译器转化成operator(cin,d1) 形参中相比<< 去掉了const
    inline istream& operator>>(istream& in, Date& d)
    {
    	in >> d._year >> d._month >> d._day;
    	return in;
    }

    Date.cpp

    #include"Date.h"
    //d1 == d2
    bool Date::operator==(const Date& d) const
    { 	return _year == d._year
    		&& _month == d._month
    		&& _day == d._day;
    }
    //d1 > d2
    bool Date::operator>(const Date& d) const
    {
    	if (_year > d._year)
    	{
    		return true;
    	}
    	else if (_year == d._year && _month > d._month)
    	{
    		return true;
    	}
    	else if (_year == d._year && _month == d._month && _day > d._day)
    	{
    		return true;
    	}
    	return false;
    }
    //d1 >= d2
    bool Date::operator>=(const Date& d) const
    {
    	return *this > d || *this == d;
    }
    //d1 <= d2
    bool Date::operator<=(const Date& d) const
    {
    	return !(*this > d);
    }
    //d1 < d2
    bool Date::operator<(const Date& d) const
    {
    	return !(*this >= d);
    }
    //d1 != d2
    bool Date::operator!=(const Date& d) const
    {
    	return !(*this == d);
    }
    //d1 += 100
    //天满了进月 月满了进年 
    Date& Date::operator+=(int day)
    {
    	//避免 d1 += -1000的情形
    	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;
    }
    //d1 + 100
    Date Date::operator+(int day) const
    {
    	Date ret(*this);
    	ret += day;//ret.operator+=(day)
    	return ret;
    }
    //d1 -= 100
    Date& Date::operator-=(int day)
    {
    	//避免 d1 -= -1000
    	if (day < 0)
    	{
    		return *this += -day;
    	}
    	_day -= day;
    	while (_day <= 0)
    	{
    		--_month;
    		if (_month == 0)
    		{
    			--_year;
    			_month = 12;
    		}
    		_day += GetMonthDay(_year, _month);
    	}
    	return *this;
    }
    //d1 - 100
    Date Date::operator-(int day) const
    {
    	Date ret(*this);
    	ret -= day;
    	return ret;
    }
    //前置++
    Date& Date::operator++()
    {
    	//会调用 operator+=(int day)
    	*this += 1;
    	return *this;
    }
    //后置++ ―多一个int参数主要是为了和前置++进行区分 构成函数重载
    Date Date::operator++(int)
    {
    	Date tmp(*this);
    	*this += 1;
    	return tmp;
    }
    //前置--
    Date& Date::operator--()
    {
    	//复用运算符重载-=
    	*this -= 1;
    	return *this;
    }
    //后置--
    Date Date::operator--(int)
    {
    	Date tmp = *this;
    	*this -= 1;
    	return tmp;
    }
    //日期-日期
    int Date::operator-(const Date& d) const
    {
    	Date max = *this;
    	Date min = d;
    	int flag = 1;
    	if (*this < d)
      //总结:凡是内部不改变成员变量 也就是不改变*this数据的 这些成员函数都应该加const
      //if (d > *this)
    	{
    		max = d;
    		min = *this;
    		flag = -1;
    	}
    	int n = 0;
    	while (min != max)
    	{
    		++n;
    		//复用++ ++到和d1日期相等 就是相差多少天
    		++min;
    	}
    	return n * flag;
    }
    //为了支持链式流插入 cout<< d1 <<d2 返回cout类对象
    //ostream& operator<<(ostream& out,const Date& d)
    //{
    //	out << d._year << "年" << d._month << "月" << d._day << "日" << endl;
    //	return out;
    //}
    关闭

    用微信“扫一扫”