日期类的实现

2023-04-20 18:58:03 浏览数 (1)

由于拆分了函数的声明和定义,所以在函数前面要加上date类,找到函数

1. 日期类的具体实现

1.查询当前月份的天数

代码语言:javascript复制
int date::getmonthday(int year, int month)
{
    int arr[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) ))
    {
        //是闰年并且是2月份
        return 29;
    }
    return arr[month];
}
  • 设置一个数组,下标对应月份,并把2月闰年的情况单拉出来

2. 构造函数的实现(注意)

代码语言:javascript复制
date::date(int year, int month, int day)
{
    //判断日期是否合法
    if ( month>0 &amp;&amp; month < 13 &amp;&amp; ( day>0&amp;&amp;day<=getmonthday(year,month) )  )
    {
        _year = year;
        _month = month;
        _day = day;
    }
    else
    {
        cout << "日期不合法" << endl;
    }
}
  • 一定要注意日期传过来是否合法存在

3.d1==d2

代码语言:javascript复制
bool date::operator==(const date&amp; d)
{
    return (_year == d._year) &amp;&amp; (_month == d._month) &amp;&amp; (_day == d._day);
}
  • 判断 年 月 日依次是否相等

4. d1!=d2

代码语言:javascript复制
bool date::operator!=(const date &amp; d)
{
    return !(*this == d);//复用d1==d2的相反逻辑
}

如果我们再次自己写就会有很多情况考虑,所以直接复用 d1==d2的取反

5. d1<d2

代码语言:javascript复制
bool date::operator<(const date&amp; d)
{
    if (_year < d._year)//年小为小
    {
        return true;
    }
    else if (_year == d._year &amp;&amp; _month < d._month)//年相等 月小为小
    {
        return true;
    }
    else if (_year == d._year &amp;&amp; _month == d._month &amp;&amp; _day < d._day)// 年月相等 天小为小
    {
        return true;
    }
    else
    {
        return false;
    }
}

年小的为小, 年相等,月小为小 年 月相等,天小为小 其他情况都为false

6. d1<=d2

代码语言:javascript复制
bool date::operator<=(const date&amp; d)
{
    return (* this < d) || (*this == d);//复用 d1<d2 和d1==d2的情况
}

d1<=d2是由 d1==d2 和d1<d2 组成,分别复用两者即可实现

7. d1>d2

代码语言:javascript复制
bool date::operator>(const date&amp; d)
{
    return !(*this < d) &amp;&amp; (*this != d);//复用 d1<d2的逻辑反 以及d1!=d2
}

d1>d2的逻辑取反是 d1<=d2 ,由于d1<d2我们已经实现过了,只需要加上d1!=d2即可

8. d1>=d2

代码语言:javascript复制
bool date::operator>=(const date&amp; d)
{
    return !(* this < d);//复用d1<d2的逻辑反 
}

复用d1<d2的逻辑取反即可实现 d1>=d2

9. 日期 =天数

代码语言:javascript复制
date&amp; date::operator =(int day)
{
    if (day < 0)// = -等价于 -=
    {
        *this -= -day;//复用-=
        return *this;
    }
    //由于是 =改变本身,所以返回*this
    int getday = _day   day;
    //判断当前加上的天数是否大于当前月份的天数
    while (getday > getmonthday(_year, _month))
    {
        getday -= getmonthday(_year, _month);
        _month  ;
        if (_month == 13)
        {
            _year  ;
            _month = 1;
        }
    }
    //最后注意剩余的getday就为当前月份的天数
    _day = getday;
    //除了作用域 *this 还在所以可以使用引用返回
    return *this;
}

当day为负时, = - 等价于-= ,所以调用-=的复用即可(-=实现在后面) 同样由于 =是对于本身操作,除了作用域还存在,所以使用引用返回

10.日期 天数

代码语言:javascript复制
date date::operator (int day)
{
    //注释的为第一种方法
    //由于不改变日期本身,所以用一个临时变量代替
    //date ret = *this;
    //int getday = ret._day   day;
    ////判断当前加上的天数是否大于当前月份的天数
    //while (getday > getmonthday(ret._year, ret._month))
    //{
    //	getday -= getmonthday(ret._year, ret._month);
    //	ret._month  ;
    //	if (ret._month == 13)
    //	{
    //		ret._year  ;
    //		ret._month = 1;
    //	}
    //}
    ////最后注意剩余的getday就为当前月份的天数
    //ret._day = getday;
    ////除了作用域 ret不在了,所以使用传值返回
    //return ret;
    //方法二
    date tmp = *this;
    tmp  = day;
    return tmp;//复用 日期 =天数的功能
}

这里用了两种方法实现,但是可以发现第一种过于繁琐,只需复用上述 =,返回临时变量tmp即可 同样由于临时变量出了作用域就不存在了,所以使用传值返回

11.日期-=天数

代码语言:javascript复制
date&amp; date::operator-=(int day)
{
    if (day < 0)
    {
        *this  = -day;//复用 =
        return *this;
    }
    _day -= day;
    //当day 小于 当前月份的天数 直接return
    //当 day大于等于 当前月份的天数 进入循环
    while (_day <= 0)
    {
        //返回到上一个月份
        _month--;
        //当月份为1时 --为0
        if (_month == 0)
        {
            _year--;
            _month = 12;
        }
        _day  = getmonthday(_year, _month);
    }
    return *this;
}

这里不太好想,如果当前日期月份的天数_day大于day,则直接return,若小于则会使_day为负进入循环,同时向上一个月借天数直到 _day<=0 若day为负,-=-等价于 = ,直接复用 =即可

12. 日期-天数

代码语言:javascript复制
date date::operator-(int day)
{
    date ret = *this;
    ret -= day;
    return ret;
}

复用上面的-=即可,返回临时变量ret

13. d和 d

代码语言:javascript复制
date&amp; date::operator  ()//  d
{
    *this  = 1;//复用 =
    return *this;
}
  • 前置 是正常调用operator ,并且返回本身,因为除了作用域还存在,所以用引用返回
代码语言:javascript复制
date date::operator  (int)//d  
{
    date ret = *this;
    *this  = 1;//复用 =
    return ret;
}

后置 因为同样是operator ,为了区分所以加上参数int(用于占位没有实际作用),构成函数重载 因为返回临时变量ret,所以用传值返回

14. --d 和 d--

代码语言:javascript复制
date&amp; date::operator--()//--d
{
    *this -= 1;//复用-=
    return *this;
}
  • 前置- -是正常调用operator- -,并且返回本身,因为除了作用域还存在,所以用引用返回
代码语言:javascript复制
date date::operator--(int)//d--
{
    date ret = *this;
    *this -= 1;//复用-=
    return ret;
}

后置- -因为同样是operator- -,为了区分所以加上参数int(用于占位没有实际作用),构成函数重载 因为返回临时变量ret,所以用传值返回

15.日期=日期 返回天数

代码语言:javascript复制
date date::operator--(int)//d--
{
    date ret = *this;
    *this -= 1;//复用-=
    return ret;
}
  int date::operator-(date&amp; d)
  {
      date max = *this;
      date min = d;
      if (*this < d)//复用<
      {
          max = d;
          min = *this;
      }
      int n = 0;
      while (min != max)//复用!=
      {
          n  ;
          if (min._day < getmonthday(min._year, min._month))
          {
              min._day  ;
          }
          else
          {
              min._month  ;
              min._day = 1;
          }
          if (min._month == 13)
          {
              min._year  ;
              min._month = 1;
          }
      }
      return n;
  }

主要使用max和min两个临时变量,使min._day不断 ,最终使max==min 结束返回计数n值

2. 函数的声明——date.h

代码语言:javascript复制
#include<iostream>
using namespace std;
class date
{
public:
    //函数声明
    int getmonthday(int year, int month);//查询当前月份的天数
    date(int year=1, int month=1, int day=1);//构造
    void print();//输出
    bool operator==(const date&amp; d);//d1==d2
    bool operator!=(const date&amp; d);//d1!=d2
    bool operator<(const date&amp; d);//d1<d2
    bool operator<=(const date&amp; d);//d1<=d2
    bool operator>(const date&amp; d);//d1>d2
    bool operator>=(const date&amp; d);//d1>=d2
    date operator (int day);//日期 天数
    date&amp; operator =(int day);//日期 =天数
    date operator-(int day);//日期-天数
    date&amp; operator-=(int day);//日期-=天数
    date&amp; operator  ();//  d
    //int 参数仅是为了占位,构成函数重载 区分前置
    date operator  (int);//d  
    date&amp; operator--();   //--d
    date operator--(int);//d--
    int  operator-( date&amp; d);//日期-日期 返回天数
private:
    int _year;
    int _month;
    int _day;
};

3. 函数的定义——date.cpp

代码语言:javascript复制
#include"date.h"
//函数实现
int date::getmonthday(int year, int month)
{
    int arr[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
    if (month == 2 &amp;&amp; (((year % 4 == 0) &amp;&amp; (year % 100 != 0)) || (year % 400 == 0) ))
    {
        //是闰年并且是2月份
        return 29;
    }
    return arr[month];
}
date::date(int year, int month, int day)
{
    //判断日期是否合法
    if ( month>0 &amp;&amp; month < 13 &amp;&amp; ( day>0&amp;&amp;day<=getmonthday(year,month) )  )
    {
        _year = year;
        _month = month;
        _day = day;
    }
    else
    {
        cout << "日期不合法" << endl;
    }

    
}
void date::print()
{
    cout << _year << "-" << _month << "-" << _day << endl;
}
bool date::operator==(const date&amp; d)
{
    return (_year == d._year) &amp;&amp; (_month == d._month) &amp;&amp; (_day == d._day);
}
bool date::operator!=(const date &amp; d)
{
    return !(*this == d);//复用d1==d2的相反逻辑
}
bool date::operator<(const date&amp; d)
{
    if (_year < d._year)//年小为小
    {
        return true;
    }
    else if (_year == d._year &amp;&amp; _month < d._month)//年相等 月小为小
    {
        return true;
    }
    else if (_year == d._year &amp;&amp; _month == d._month &amp;&amp; _day < d._day)// 年月相等 天小为小
    {
        return true;
    }
    else
    {
        return false;
    }
}
bool date::operator<=(const date&amp; d)
{
    return (* this < d) || (*this == d);//复用 d1<d2 和d1==d2的情况
}
bool date::operator>(const date&amp; d)
{
    return !(*this < d) &amp;&amp; (*this != d);//复用 d1<d2的逻辑反 以及d1!=d2
}
bool date::operator>=(const date&amp; d)
{
    return !(* this < d);//复用d1<d2的逻辑反 
}

date&amp; date::operator =(int day)
{
    if (day < 0)// = -等价于 -=
    {
        *this -= -day;//复用-=
        return *this;
    }
    //由于是 =改变本身,所以返回*this
    int getday = _day   day;
    //判断当前加上的天数是否大于当前月份的天数
    while (getday > getmonthday(_year, _month))
    {
        getday -= getmonthday(_year, _month);
        _month  ;
        if (_month == 13)
        {
            _year  ;
            _month = 1;
        }
    }
    //最后注意剩余的getday就为当前月份的天数
    _day = getday;
    //除了作用域 *this 还在所以可以使用引用返回
    return *this;
}
date date::operator (int day)
{
    ////由于不改变日期本身,所以用一个临时变量代替
    //date ret = *this;
    //int getday = ret._day   day;
    ////判断当前加上的天数是否大于当前月份的天数
    //while (getday > getmonthday(ret._year, ret._month))
    //{
    //	getday -= getmonthday(ret._year, ret._month);
    //	ret._month  ;
    //	if (ret._month == 13)
    //	{
    //		ret._year  ;
    //		ret._month = 1;
    //	}
    //}
    ////最后注意剩余的getday就为当前月份的天数
    //ret._day = getday;
    ////除了作用域 ret不在了,所以使用传值返回
    //return ret;
    date tmp = *this;
    tmp  = day;
    return tmp;//复用 日期 =天数的功能
}


date&amp; date::operator-=(int day)
{
    if (day < 0)
    {
        *this  = -day;//复用 =
        return *this;	
    }
    _day -= day;
    //当day 小于 当前月份的天数 直接return
    //当 day大于等于 当前月份的天数 进入循环
    while (_day <= 0)
    {
        //返回到上一个月份
        _month--;
        //当月份为1时 --为0
        if (_month == 0)
        {
            _year--;
            _month = 12;
        }
        _day  = getmonthday(_year, _month);
    }
    return *this;
}
date date::operator-(int day)
{
    date ret = *this;
    ret -= day;
    return ret;
}
date&amp; date::operator  ()//  d
{
    *this  = 1;
    return *this;
}
date date::operator  (int)//d  
{
    date ret = *this;
    *this  = 1;
    return ret;
}

date&amp; date::operator--()//--d
{
    *this -= 1;//复用-=
    return *this;
}

date date::operator--(int)//d--
{
    date ret = *this;
    *this -= 1;//复用-=
    return ret;
}
  int date::operator-(date&amp; d)
  {
      date max = *this;
      date min = d;
      if (*this < d)
      {
          max = d;
          min = *this;
      }
      int n = 0;
      while (min != max)
      {
          n  ;
          if (min._day < getmonthday(min._year, min._month))
          {
              min._day  ;
          }
          else
          {
              min._month  ;
              min._day = 1;
          }
          if (min._month == 13)
          {
              min._year  ;
              min._month = 1;
          }
      }
      return n;
  }

4.函数调用——main.cpp

代码语言:javascript复制
#include"date.h"
int main()
{
    date d1(2023, 2, 9);
    date d2(2023, 2, 1);
    cout << (d1==d2) << endl;
    cout << (d1 < d2) << endl;
    cout << (d1 - d2) << endl;
    return 0;
}

0 人点赞