基于C++实现简单的日期计算机

来自:网络
时间:2024-06-09
阅读:

引言

我们日常生活中可能会有一个烦恼。

今天几月几号?过n天后又是几月几号?某年某月某天和x年x月x天相差几天?你和男/女朋友的相识了几天?等等。这些问题好麻烦,我不想去算,所以我们的日期计算机也就油然而生了。

头文件的准备

头文件的声明代码:

#pragma once
#include<iostream>
#inclu<assert.h>
using namespace std;
class Date
{
public:
    // 获取某年某月的天数
    int GetMonthDay(int year, int month)
    {
        static int days[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31,30,31 };
        int day = days[month];
        if (month == 2 &&
            (year & 4 == 0 && year % 100 != 0) || (year % 400 == 0))
        {
            day += 1;
        }
        return day;
    }
    //打印函数
    void Print()
    {
        cout << _year << "-" << _month << "-" << _day << endl;
    }
    // 全缺省的构造函数
    Date(int year = 1900, int month = 1, int day = 1);
    // 拷贝构造函数
  // d2(d1)
    Date(const Date& d);
    // 赋值运算符重载
  // d2 = d3 -> d2.operator=(&d2, d3)
    Date& operator=(const Date& d);
    // 析构函数
    ~Date();
    // 日期+=天数
    Date& operator+=(int day);
    // 日期+天数
    Date operator+(int day);
    // 日期-天数
    Date operator-(int day);
    // 日期-=天数
    Date& operator-=(int day);
    // 前置++
    Date& operator++();
    // 后置++
    Date operator++(int);
    // 后置--
    Date operator--(int);
    // 前置--
    Date& operator--();
    // >运算符重载
    bool operator>(const Date& d);
    // ==运算符重载
    bool operator==(const Date& d);
    // >=运算符重载
    bool operator >= (const Date& d);
    // <运算符重载
    bool operator < (const Date& d);
    // <=运算符重载
    bool operator <= (const Date& d);
    // !=运算符重载
    bool operator != (const Date& d);
    // 日期-日期 返回天数
    int operator-(const Date& d);
private:
    int _year;
    int _month;
    int _day;
};

函数代码的实现

1.某年某月天数的获取

第一个函数,我们定义在了头文件中

// 获取某年某月的天数
int GetMonthDay(int year, int month)
{
    assert(month > 0 && month < 13);
    static int DayA[13] = { 0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31,30,31 };
    int day = DayA[month];
    if (month == 2 &&
        (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0))
    {
        day += 1;
    }
    return day;
}

这里我们直接建立一个数组,存储的是每月的天数。

第一个我们存储为0,毕竟第一个月我们没有经历满。

这里的数组我们前边加上了static,使其成为了全局变量。这里设置为全局变量是为了方便后边的使用/调用。

不要忽略了这里闰年的判定,最后返回我们的天数

2.全缺省的构造函数

Date::Date(int year,int month,int day)
{
    year = _year;
    month = _month;
    day = _day;
}

这个函数就比较简单了,就是赋值。

值得注意的是:这里是源文件,我们在头文件声明是已经使用了缺省参数,所以在源文件中就不需要使用了。

3.拷贝构造函数

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

这个函数也比较简单,我们这里利用了this指针,所以传参只需要传递一个就可以了。也不过多讲解了。

4.七个运算符的重载

首先是==的重载

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

然后再写一个 < 的重载

bool Date::operator<(const Date& d)
{
    if (_year < d._year)
    {
        return true;
    }
    else if (_year == d._year)
    {
        if (_month < d._month)
            return true;
        else if (_month == d._month)
        {
            if (_day < d._day)
            {
                return true;
            }
        }
    }
    return false;
}

写完这两个后,我们后边其他的重载就好写了。

这里我们不需要再去写这种麻烦的代码,我们偷个懒,转换一下思路。利用前边这两个去写其他的重载函数

小于等于就是符合小于的同时也符合等于

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

大于就是不是小于等于 

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

大于等于就是不是小于 

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

不等于就是不是等于就可以 

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

赋值运算符重载

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

这个我们在之前也讲解过,就不过多叙述了

5.四个日期计算函数

第一个:月份日期增加天数

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

首先先获取天数

如果天数大于月份的天数,我们就减去当前月份的天数,然后让月份加一。如果月份大于13了,我们就让年去加一。

写完+=,那么+就会好写很多

Date Date::operator+(int day)
{
    Date ret = *this;
    ret += day;
    return  ret;
}

这里我们也用到了+=的重载。

但这两个都是用来加天数的,两者有什么不同呢?

首先呢,第一个重载的是运算符+=的重载,他们的本质是在原有的基础上进行改变,改变了*this指针的值

举个例子

int main()
{
    Date d1(2024, 4, 24);
    d1 += 3;//这里的d1应该是2024/4/27
    d1.Print();
    return 0;
}

基于C++实现简单的日期计算机

利用+=我们改变了d1原本的数值

再看一看+

基于C++实现简单的日期计算机

虽然d1加了3,但是并没有改变d1原本的数值

总结:

  • Date& Date::operator+=(int day)  他的返回类型是Date的引用们可以进行链式操作,调用+=,是在原有的基础上进行增加天数,会改变原有变量的数值。
  • Date Date::operator+(int day) 他的返回类型是一个新的Date,需要去利用变量去接受。调用+,返还的对象实在原有的基础上进行增加得到结果,不会改变原有变量的数值

那么问题又来了?

我这里显示写出+=,然后再+中去调用+=。

那么如果我们先写出+后,再重在+=中去调用+会怎么样呢???

这里就直接说结论了 

不论是先写+=再写+,还是先写+再写+=,结果都是一样的,但是唯一不同的就是代码的书写量以及效率的高低。在这里我比较推荐我的写法,先写+=,再写+会提高一定的效率。

 写完+有关的,我们开始写-有关的

接下来是月份日期的减少

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

6.前置/后置的加加减减

这里需要注意的是,为了区分时前置和后置的区别,我们在传参时需要加入一个int

加入int参数的就是后置++

这里代码如下:

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

既然是加加就是day加一,这样的话我们也利用前边思路想法,前置我们利用引用去写,而后置我们不用引用,利用前边的运算符重载去写

7.计算两个日期之间的天数值

int Date::operator-(const Date& d)
{
    int flag = 1;
    Date max = *this;
    Date min = d;
 
    if (*this < d)
    {
        int flag = -1;
        max = d;
        min = *this;
    }
 
    int n = 0;
    while (min != max)
    {
        ++min;
        ++n;
    }
 
    return n * flag;
}

这个函数的逻辑也利用了假设法,其思路则是确定好大的日期后,让小的日期不断增加然后直至与大的日期相同。

讲解:

第一个点是确确定日期的大小。我们让max确保成为大日期,min确保成为小日期,这里就利用了假设法,设置完后,利用if语句,确保我们的判断是正确的,如果不正确就进行反转,并让flag成为-1。

第二个点就是计算两者之间差值的天数。我们通过循环,每次都让min不断增加,同时也见建立一个变量n,让它随着min增加,从而记录下两者之间的差值天数,最后n就是两者之间的差值。

第三个点就是返回值。我们这里的返回值最后需要乘上flag,其原因是,我们一开始我们让this指针为大日期,如果不对的话,就说明this指针的日期小,所以两者之间差值就是负的。

以上就是基于C++实现简单的日期计算机的详细内容,更多关于C++日期计算机的资料请关注其它相关文章!

返回顶部
顶部