C++日期类运算符重载方式

 更新时间:2023年02月06日 08:31:51   作者:安河桥畔  
这篇文章主要介绍了C++日期类运算符重载方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教

构造函数

定义一个全缺省的构造函数,函数体内判断当前日期是否合法,如果是非法日期,则初始化为2022,1,1

Date(int year=2022, int month=1, int day=1)
        :_year(year)
        , _month(month)
        , _day(day)
    {
        if (year <1 || month < 1 || month>12 || day<1 || day>GetDayOfMonth(year,month))
        {
            _year = 2022;
            _month = 1;
            _day = 1;
        }
    }

拷贝构造

    Date(const Date& d)
    {
        _year = d._year;
        _month = d._month;
        _day = d._day;
    }

赋值运算符重载

赋值运算符编译器会默认生成,对于日期类可以不用自己定义,自己定义需要注意三点:

  • 参数和返回值都是类类型的引用
  • 判断是否为自己给自己赋值,因为两对象不能直接比较,所以判断时用地址
  • 为了支持连续赋值,必须返回*this
    Date& operator=(const Date& d)
    {
        //不能用对象直接比较,只能比较地址
        //if (d != (*this))
        if(&d!=this)
        {
            _year = d._year;
            _month = d._month;
            _day = d._day;
        }
        return *this;
    }

“+” 日期+天数

"+“与”+=“区分,”+"不改变原变量的值,只是通过返回值返回,所以这里要定义一个临时变量temp,由于temp是保存在栈上的临时变量,所以返回值为Date类型,不能返回引用

先判断正负,如果加的数是负数,则复用"-"的重载

加完后判断_day的合法性,如果_day非法,_day减去当前月份的天数,月份增加,直到日期合法

    //日期+天数
    Date operator+(int day)
    {
        //不能改变原有日期,所以要定义临时变量
        //临时变量不能返回引用
        if (day < 0)
        {
            day = -day;
            return *this - day;
        }
        Date temp=*this;
        temp._day += day;
        while (temp._day > GetDayOfMonth(_year, _month))
        {
            temp._day -= GetDayOfMonth(temp._year,temp._month);
            temp._month++;
            if (temp._month > 12)
            {
                temp._month = 1;
                temp._year++;
            }
        }
        return temp;
    }

“-” 日期-天数

与"+"同原理,如果_day减为0或者负数,先将月份下调一个月,再将日加上上调后月份的天数,如8月0日表示7月31,8月-1日,则表示7月30日。

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

“+=” 日期+=天数

"+=“与”+“的区别是,加后要个自己赋值,所以返回类类型对象的引用,可以复用”=“和”+"的重载

    //日期+=天数
    Date& operator+=(int days)
    {
        //复用=和+
        //这里要先将加的和赋值给*this,再返回
        //非常量左值引用只能引用非常量左值
        //如果要直接return(*this + days),返回值类型应为Date&&
        *this = *this + days;
        return * this;
    }

“-=” 日期-=天数

    //日期-=天数
    Date& operator-=(int days)
    {
        *this = *this - days;
        return *this;
    }

前置"++"

前置"++"返回的是加一后的值,可以直接给*this加一后返回

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

后置"++"

C++中,后置"++“参数加一个int,无实际意义,区分前置与后置”++“,由于后置”++"要返回的是自增之前的值,所以要定义一个临时变量,自增后返回该临时变量的值

    //后置++
    //
    Date operator++(int)
    {
        //返回的是自增之前的值
        Date temp = *this;
        *this += 1;
        return temp;
    }

前置"–"

与前置"++"原理相同

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

后置"- -"

与后置"++"原理相同

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

“==”

判断每变量是否相等

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

“>”

从年到月日依次判断

    //>
    bool operator>(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;
        }
        else
        {
            return false;
        }
    }

“<”

    //<
    bool operator<(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;
        }
        else
        {
            return false;
        }
    }

“-” 日期-日期

定义一个计数器统计从小的日期到大的日期需要自增多少次

    int operator-(const Date& d)const
    {
        int count = 0;
        Date start(*this);
        Date end(d);
        //保证大的日期减小的日期
        if (start>end)
        {
            Date temp = end;
            end = start;
            start = temp;
        }
        while(end>start)
        {
            start++;
            count++;
        }
        return count;
    }

设置年、月、日

对外部提供修改对象值的接口

    void SetYear(int year)
    {
        _year = year;
    }

    void SetMonth(int month)
    {
        _month = month;
    }

    void SetDay(int day)
    {
        _day = day;
    }

获取年、月、日

向外部提供获取类成员变量值的接口

    int GetYear()const
    {
        return _year;
    }

    int GetMonth()const
    {
        return _month;
    }
    
    int GetDay()const
    {
        return _day;
    }

判断闰年

判断闰年,可以定义为private,不用暴露给外部

    bool IsLeapYear(int year)
    {
        if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0)
        {
            return true;
        }
        return false;
    }

获取当前月天数

定义一个数组保存每月的天数,根据传进来的年和月返回当前月份的天数

    //获取当前月份的天数
    int GetDayOfMonth(int year, int month)
    {
        int arr[] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
        if (IsLeapYear(year))
        {
            arr[2] = 29;
        }
        return arr[month];
    }

日期类完整代码

class Date
{
public:
    //全缺省构造函数
    Date(int year=2022, int month=2, int day=2)
        :_year(year)
        , _month(month)
        , _day(day)
    {
        if (year <1 || month < 1 || month>12 || day<1 || day>GetDayOfMonth(year,month))
        {
            _year = 2022;
            _month = 1;
            _day = 1;
        }
    }
    //拷贝构造
    Date(const Date& d)
    {
        _year = d._year;
        _month = d._month;
        _day = d._day;
    }
    //析构——无实际意义
    ~Date()
    {
        cout << "~Date()" << endl;
    }

    //赋值运算符重载(不写也可以,编译器会默认生成)
    Date& operator=(const Date& d)
    {
        //不能用对象直接比较,只能比较地址
        //if (d != (*this))
        if(&d!=this)
        {
            _year = d._year;
            _month = d._month;
            _day = d._day;
        }
        return *this;
    }

    //日期+天数
    Date operator+(int day)
    {
        //不能改变原有日期,所以要定义临时变量
        //临时变量不能返回引用
        if (day < 0)
        {
            day = -day;
            return *this - day;
        }
        Date temp=*this;
        temp._day += day;
        while (temp._day > GetDayOfMonth(_year, _month))
        {
            temp._day -= GetDayOfMonth(temp._year,temp._month);
            temp._month++;
            if (temp._month > 12)
            {
                temp._month = 1;
                temp._year++;
            }
        }
        return temp;
    }

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

    //日期+=天数
    Date& operator+=(int days)
    {
        //复用=和+
        //这里要先将加的和赋值给*this,再返回
        //非常量左值引用只能引用非常量左值
        //如果要直接return(*this + days),返回值类型应为Date&&
        *this = *this + days;
        return * this;
    }

    //日期-=天数
    Date& operator-=(int days)
    {
        *this = *this - days;
        return *this;
    }

    //前置++
    Date& operator++()
    {
        *this += 1;
        return *this;
    }
    //后置++
    //C++语法,后置++参数加一个int,无实际意义,区分前置与后置++
    Date operator++(int)
    {
        //返回的是自增之前的值
        Date temp = *this;
        *this += 1;
        return temp;
    }

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

    //后置--
    Date operator--(int)
    {
        Date temp = *this;
        *this-= 1;
        return temp;
    }
    //==
    bool operator==(Date& d)const
    {
        return d._year == _year &&
            d._month == _month &&
            d._day == _day;
    }

    //>
    bool operator>(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;
        }
        else
        {
            return false;
        }
    }

    //<
    bool operator<(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;
        }
        else
        {
            return false;
        }
    }
    //日期-日期 即还有多少天到某一天 
    int operator-(const Date& d)const
    {
        int count = 0;
        Date start(*this);
        Date end(d);
        //保证大的日期减小的日期
        if (start>end)
        {
            Date temp = end;
            end = start;
            start = temp;
        }
        while(end>start)
        {
            start++;
            count++;
        }
        return count;
    }

    //设置年、月、日接口
    void SetYear(int year)
    {
        _year = year;
    }

    void SetMonth(int month)
    {
        _month = month;
    }

    void SetDay(int day)
    {
        _day = day;
    }

    //获取年、月、日接口
    int GetYear()const
    {
        return _year;
    }

    int GetMonth()const
    {
        return _month;
    }
    
    int GetDay()const
    {
        return _day;
    }
private:
    int _year;
    int _month;
    int _day;
    //判断闰年
    bool IsLeapYear(int year)
    {
        if ((year % 4 == 0 && year % 100 != 0) || year % 400 == 0)
        {
            return true;
        }
        return false;
    }

    //获取当前月份的天数
    int GetDayOfMonth(int year, int month)
    {
        int arr[] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
        if (IsLeapYear(year))
        {
            arr[2] = 29;
        }
        return arr[month];
    }
};

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • Qt实现转动轮播图

    Qt实现转动轮播图

    这篇文章主要为大家详细介绍了Qt实现转动轮播图,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-06-06
  • C程序和C++程序之间的互相调用图文教程

    C程序和C++程序之间的互相调用图文教程

    这篇文章主要给大家介绍了关于C程序和C++程序之间互相调用的相关资料,我们平常在刷题的时候,难免遇到实现多组输入这样的问题,文中通过图文介绍的非常详细,需要的朋友可以参考下
    2023-07-07
  • 嵌入式QT移植的实现

    嵌入式QT移植的实现

    本文主要介绍了嵌入式QT移植的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • C++基于CMD命令行实现扫雷小游戏

    C++基于CMD命令行实现扫雷小游戏

    这篇文章主要为大家详细介绍了C++基于CMD命令行实现扫雷小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-05-05
  • 用C语言实现简单五子棋小游戏

    用C语言实现简单五子棋小游戏

    这篇文章主要为大家详细介绍了用C语言实现简单五子棋小游戏,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-07-07
  • C语言编程PAT乙级学习笔记示例分享

    C语言编程PAT乙级学习笔记示例分享

    这篇文章主要为大家介绍了C语言编程PAT乙级学习笔记实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-05-05
  • 解决codeblocks断点不停无效的问题

    解决codeblocks断点不停无效的问题

    今天小编就为大家分享一篇解决codeblocks断点不停无效的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2019-12-12
  • C++类静态成员与类静态成员函数详解

    C++类静态成员与类静态成员函数详解

    静态成员不可在类体内进行赋值,因为它是被所有该类的对象所共享的。你在一个对象里给它赋值,其他对象里的该成员也会发生变化。为了避免混乱,所以不可在类体内进行赋值
    2013-09-09
  • Assert(断言实现机制深入剖析)

    Assert(断言实现机制深入剖析)

    言前后最好空一格[编程风格的问题,按你自已的喜好,适合自已就最好]。断言只是用来检查程序的逻辑正确性,不能代替条件替换。断言比printf语句这种形式的打印好使
    2013-09-09
  • C语言超详细文件操作基础上篇

    C语言超详细文件操作基础上篇

    这篇文章主要为大家详细介绍了C语言的文件操作,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下,希望能够给你带来帮助
    2022-03-03

最新评论