C++获取下一天和上一天

根据今天的日期获取下一天的日期,输入日期格式是20190520 的字符串

方便自己和别人使用

int is_leap_year(int year)
{
    return year % 400 == 0 || (year % 4 == 0 && year % 100 != 0);
}
std::string get_next_day(const std::string& cur_day)
{
    if (cur_day.size() < 8) {
        return std::string("");
    }
    int year = 0, month = 0, day = 0;
    try {
        year = std::stoi(cur_day.substr(0, 4));
        month = std::stoi(cur_day.substr(4, 2));
        day = std::stoi(cur_day.substr(6, 2));
    }
    catch (std::invalid_argument) {
        return std::string("");
    }
    if (12 == month) {
        if (31 == day) {
            year += 1; month = 1; day = 1;
        }
        else if (day < 31) {
            day += 1;
        }
        else {
            return std::string("");
        }
        char buf[64] = {0};
        std::snprintf(buf, sizeof(buf), "%04d%02d%02d", year, month, day);
        return std::string(buf);
    }
    bool is_leap = is_leap_year(year);
    ///< 处理2月
    if (2 == month) {
        if (is_leap) {
            if (29 == day) {
                day = 1;
                month = 3;
            }
            else if (day < 29) {
                day += 1;
            }
            else {
                return std::string("");
            }
        }
        else {
            if (28 == day) {
                day = 1;
                month = 3;
            }
            else if (day < 28) {
                day += 1;
            }
            else {
                return std::string("");
            }
        }
        char buf[64] = {0};
        std::snprintf(buf, sizeof(buf), "%04d%02d%02d", year, month, day);
        return std::string(buf);
    }
    switch (month) {
        case 1:
        case 3:
        case 5:
        case 7:
        case 8:
        case 10:
        {
            if (31 == day) {
                month += 1;
                day = 1;
            }
            else if (day < 31) {
                day += 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 4:
        case 6:
        case 9:
        case 11:
        {
            if (30 == day) {
                month += 1;
                day = 1;
            }
            else if (day < 30) {
                day += 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        default:
        {
            return std::string("");
        }
        break;
    }
    char buf[64] = {0};
    std::snprintf(buf, sizeof(buf), "%04d%02d%02d", year, month, day);
    return std::string(buf);
}

下边是获取上一天的代码

std::string get_last_day(const std::string& cur_day)
{
    if (cur_day.size() < 8) {
        return std::string("");
    }
    int year = 0, month = 0, day = 0;
    try {
        year = std::stoi(cur_day.substr(0, 4));
        month = std::stoi(cur_day.substr(4, 2));
        day = std::stoi(cur_day.substr(6, 2));
    }
    catch (std::invalid_argument) {
        return std::string("");
    }
    int limit_day = is_leap_year(year) ? 29 : 28;
    switch (month) {
        case 1:
        {
            if (1 == day) {
                year -= 1; month = 12; day = 31;
            }
            else if (day <= 31) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 3:
        {
            if (1 == day) {
                month = 2; day = limit_day;
            }
            else if (day <= 31) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 12:
        case 10:
        case 7:
        case 5:
        {
            if (1 == day) {
                day = 30; month -= 1;
            }
            else if (day <= 31) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 8:
        {
            if (1 == day) {
                day = 31; month = 7;
            }
            else if (day <= 31) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 2:
        {
            if (1 == day) {
                day = 31; month = 1;
            }
            else if (day <= limit_day) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        case 4:
        case 6:
        case 9:
        case 11:
        {
            if (1 == day) {
                day = 31; month -= 1;
            }
            else if (day <= 30) {
                day -= 1;
            }
            else {
                return std::string("");
            }
        }
        break;
        default:
        {
            return std::string("");
        }
        break;
    }
    char buf[64] = {0};
    std::snprintf(buf, sizeof(buf), "%04d%02d%02d", year, month, day);
    return std::string(buf);
}

简化一下写法,上边的有不少重复代码

std::string get_next_day_op(const std::string& cur_day)
{
    if (cur_day.size() < 8) {
        return std::string("");
    }
    int year = 0, month = 0, day = 0;
    try {
        year = std::stoi(cur_day.substr(0, 4));
        month = std::stoi(cur_day.substr(4, 2));
        day = std::stoi(cur_day.substr(6, 2));
    }
    catch (std::invalid_argument) {
        return std::string("");
    }
    int year_addend = 0, day_limit = 0;
    switch (month) {
        case 12:
        {
            year_addend = 1; day_limit = 31;
        }
        break;
        case 1:
        case 3:
        case 5:
        case 7:
        case 8:
        case 10:
        {
            year_addend = 0; day_limit = 31;
        }
        break;
        case 4:
        case 6:
        case 9:
        case 11:
        {
            year_addend = 0; day_limit = 30;
        }
        break;
        case 2:
        {
            year_addend = 0;
            day_limit = is_leap_year(year) ? 29 : 28;
        }
        break;
        default:
        {
            return std::string("");
        }
        break;
    }

    if (day_limit == day) {
        year += year_addend;
        month += 1;
        if (13 == month) {
            month = 1;
        }
        day = 1;
    }
    else if (day < day_limit) {
        day += 1;
    }
    else {
        return std::string("");
    }
    char buf[64] = {0};
    std::snprintf(buf, sizeof(buf), "%04d%02d%02d", year, month, day);
    return std::string(buf);
}

当然还可以用时间戳转换的方式来进行,这样就不存在针对不同月份和闰年的处理了,不过效率会低一些,注意这个函数没有处理输入月份和输入天数异常的情况,比如输入天数超过31天之类的问题


const uint32_t ONE_DAY_SECS = 24 * 60 * 60;
std::string get_next_day_by_timestamp(const std::string& cur_day)
{
    if (cur_day.size() < 8) {
        return std::string("");
    }
    int year = 0, month = 0, day = 0;
    try {
        year = std::stoi(cur_day.substr(0, 4));
        month = std::stoi(cur_day.substr(4, 2));
        day = std::stoi(cur_day.substr(6, 2));
    }
    catch (std::invalid_argument) {
        return std::string("");
    }
    struct tm t;
    t.tm_year = year - 1900;
    t.tm_mon = month - 1;
    t.tm_mday = day;
    t.tm_hour = 1; t.tm_min = 0; t.tm_sec = 0;
    uint64_t next_day_timestamp = std::mktime(&t) + ONE_DAY_SECS;
    struct tm time_info = *std::localtime((std::time_t*)&next_day_timestamp);
    char buf[64] = {0};
    std::strftime(buf, sizeof(buf), "%Y%m%d", &time_info);
    return std::string(buf);
}

 

  • 1
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值