Определение разницы между датами в днях

HardLogin

Всем привет! Задача в том что есть две даты ( день, месяц, год ) нужно определить разницу дв днях между ними)
5 ответов

HardLogin

Это я давно когда-то делал. Проверял в подобном он-лайн калькуляторе. Работает вроде правильно. Конструктор Date принимает три параметра: год, месяц и день. Перегружены операторы -, +=, -=, =. Т.е. ты можешь создать два объекта Date, причем первая дата должна быть меньше второй, иначе оно неправильный результат возвращает. Но то можно исправить. Ну и разница этих дат вернет количество дней.
class Date
{
    int y;
    int m;
    int d;
public:
    //конструктор
    Date(int y = 0, int m = 0, int d = 0)
    {
        cout << "Вызов конструктора\n";
        this->y = y;
        this->m = m;
        this->d = d;
    }
    //деструктор
    ~Date() {cout << "Вызов деструктора\n"; }
 
    int GetY() const { return y; }
    int GetM() const { return m; }
    int GetD() const { return d; }
    const char* GetDate(char *str, const char *ch=".") const;       //возвращает дату как строку
    //по умолчанию разделитель между месяцем днем и годом "."
    //можно задать свой
 
    void SetDate(int y, int m, int d) 
    {
        this->y = y;
        this->m = m;
        this->d = d;
    }
    void SetY(int y) { this->y = y; }
    void SetM(int m) { this->m = m; }
    void SetD(int d) { this->d = d; }
 
    void operator+=(int days);          //прибавляет к дате N дней
    void operator-=(int days);          //отнимает от даты N дней
    int operator-(const Date &date);    //возвращает разницу в днях между двумя датами
    void operator=(const Date &date);   //приравнивает одну дату другой 
};
 
const char* Date::GetDate(char *str, const char *ch) const
{
    char *buf = new char[10];
    if(d<10) strcat(strcat(strcpy(str, "0"), itoa(d,buf,10)), ch);
    else strcat(strcpy(str, itoa(d,buf,10)), ch);
    if(m<10) strcat(strcat(strcat(str, "0"), itoa(m,buf,10)), ch);
    else strcat(strcat(str, itoa(m,buf,10)), ch);
    strcat(str, itoa(y,buf,10));
    delete [] buf;
    return str;
}
void Date::operator+=(int days)
{
    //число дней по месяцам в году (невисокосном)
    int month_days[12] = {31,28,31,30,31,30,31,31,30,31,30,31};
    if((!(y%4) && y%100) || (!(y%100) && !(y%400))) month_days[1] = 29;
    //вычисляем сколько до дней до конца месяца, точнее вычитаем этот остаток от days
    if(days <= month_days[m-1]-d)
        d+=days;
    else
        {           
            days -= month_days[m-1] - d+1;
            d=1;
            if(m<11)
                m++;
            else 
                {
                    m=1;
                    y++;
                    if((!(y%4) && y%100) || (!(y%100) && !(y%400))) month_days[1] = 29;
                    else  month_days[1] = 28;
                }
            while(days)
            {
                for(m; days>=month_days[m-1];)
                    if(days >= month_days[m-1])
                    {
                        days -= month_days[m-1];
                        m++;
                        if(m==13)
                        {
                            m=1;
                            y++;
                            if((!(y%4) && y%100) || (!(y%100) && !(y%400))) month_days[1] = 29;
                            else  month_days[1] = 28;
                        }                       
                    }else
                    {
                        d+=days;
                        days=0; 
                        break;
                    }   
                    if(days)
                    {
                        d+=days;
                        days=0; 
                    }
            }
        }   
    
}
int Date::operator-(const Date &date) {
    //число дней по месяцам в году (невисокосном)
    int month_days[12] = {31,28,31,30,31,30,31,31,30,31,30,31};
    //если обе даты равны - возвращаем "0"
    if(y == date.GetY() && m == date.GetM() && d == date.GetD()) return 0;
 
    if(y == date.GetY())
        if(m == date.GetM())
            return date.GetD() - d;
        else /*if(m != date.GateM())*/
        {
            int days = 0;
            //если текущий год високосный - устанавливаем в феврале 28 дней
            if((!(y%4) && y%100) || (!(y%100) && !(y%400))) month_days[1] = 29;
            //получаем остаток дней до конца месяца;
            days = month_days[m-1] - d; 
            //суммируем дни "целых" месяцов
            for(int i = m; i < date.GetM()-1; i++)
                days += month_days[i];
            //прибавляем оставшие дни и возвращаем значение в вызывающую ф-ю
            return days += date.GetD();
        }
    else /*if(y != date.GateY())*/
    {
        int days = 0;
        //сначала вычисляем число дней до конца текущего года
        //если начальный год високосный - устанавливаем в феврале 28 дней
        if((!(y%4) && y%100) || (!(y%100) && !(y%400))) month_days[1] = 29;
        //плюсуем остаток дней до конца месяца
        days = month_days[m-1] - d;
        //далее плюсуем число дней каждого месяца до конца года
        for(int i = m; i<12; i++)
            days += month_days[i];
        //потом плюсуем дни по годам (365 и 366 в високосном)
        for(int i = y+1; i < date.GetY(); i++)
            if((!(i%4) && i%100) || (!(i%100) && !(i%400))) days+=366;
            else days+=365; 
        //если последний год високосный - устанавливаем в феврале 28 дней
        //иначе - 30        
        if((!(date.GetY()%4) && date.GetY()%100) || (!(date.GetY()%100) && !(date.GetY()%400)))  month_days[1] = 29;
        else month_days[1] = 28;
        //плюсуем число дней "целых" месяцев
        for(int i = 0; i < date.GetM()-1; i++)
            days += month_days[i];
        //плюсуем оставшиеся дний
        return days += date.GetD();
    }
}
void Date::operator=(const Date &date)
{
    d = date.GetD();
    m = date.GetM();
    y = date.GetY();
}


HardLogin

я не понял как вы определили высокосный или нет год. if((!(y%4) && y%100) || (!(y%100) && !(y%400))) почему просто не !y % 4?


HardLogin

HardLogin, если имеется boost, то можно так:
#include <iostream>
#include <boost/date_time/gregorian/gregorian.hpp>
 
int main() 
{
    using namespace boost::gregorian;
    
    date begin(2000, Feb, 10);
    date end(2005, Aug, 19);
    date_period period(begin, end);
    std::cout<<"days in period: "<<period.length()<<"\n";
    
    return 0;
}


HardLogin


HardLogin

так все таки почему именно так определяют высокосный год) может я не знаю что такое высокосный год) а так все понятно