牛客网[NOIP2016]回文日期

在日常生活中,通过年、月、日这三个要素可以表示出一个唯一确定的日期。

牛牛习惯用8位数字表示一个日期,其中,前4位代表年份,接下来2位代表月份,最后2位代表日期。显然:一个日期只有一种表示方法,而两个不同的日期的表示方法不会相同。

牛牛认为,一个日期是回文的,当且仅当表示这个日期的8位数字是回文的。现 在,牛牛想知道:在他指定的两个日期之间包含这两个日期本身),有多少个真实存在的日期是回文的。

一个8位数字是回文的,当且仅当对于所有的i ( 1 <=i<= 8 )从左向右数的第i个 数字和第9-i个数字(即从右向左数的第i个数字)是相同的。

例如:

•对于2016年11月19日,用8位数字20161119表示,它不是回文的。

•对于2010年1月2日,用8位数字20100102表示,它是回文的。

•对于2010年10月2日,用8位数字20101002表示,它不是回文的。

每一年中都有12个月份:

其中,1、3、5、7、8、10、12月每个月有31天;4、6、9、11月每个月有30天;而对于2月,闰年时有29天,平年时有28天。

一个年份是闰年当且仅当它满足下列两种情况其中的一种:

1.这个年份是4的整数倍,但不是100的整数倍;

2.这个年份是400的整数倍。

例如:

•以下几个年份都是闰年:2000、2012、2016。

•以下几个年份是平年:1900、2011、2014。

输入描述:

输入包括两行,每行包括一个8位数字。
第一行表示牛牛指定的起始日期。
第二行表示牛牛指定的终止日期。
保证datei和都是真实存在的日期,且年份部分一定为4位数字,且首位数字不为0。
保证date1 —定不晚于date2。

输出描述:

输出一行,包含一个整数,表示在date1和date2之间,有多少个日期是回文的。

———————————————————————————————————————————

很显然,我们不会选择去一个一个遍历这两个日期之间的所有日期,去判断它们是否是回文日期,因为一年中回文日期最多只有一个,很多年份甚至没有回文日期。所以我们的思路应该是先拿到一个回文日期,然后再去判断它是否在给定时间区间内。拿到一个回文日期可以从年份着手,也可以从月份日期着手。

解法一(年份):

首先拿到起始时间和结束时间的年份,遍历这两个年份之间的所有年份,求它们对应的回文日期,看这个日期存在是否合理,如果合理再看这个日期是否在给定的时间区间内。

#include<iostream>
#include<string>
using namespace std;
int month[13] = { 0,31,28,31,30,31,30,31,31,30,31,30,31 };
string d1, d2;
int sm, sd, em, ed, sy, ey, i, m, d, sum = 0;//m、d是回文年份对应的月份、日期
//判断是否在起始时间的后面
bool hous() {
    if (m < sm) {
        return false;
    }
    if (m == sm) {
        if (d < sd) {
            return false;
        }
    }
    return true;
}
//判断是否在结束时间的前面
bool qiane() {
    if (m > em) {
        return false;
    }
    if (m == em) {
        if (d > ed) {
            return false;
        }
    }
    return true;
}
int main() {
    ios::sync_with_stdio(false);
    cin.tie(0);cout.tie(0);
    cin >> d1 >> d2;
    sy = (d1[0] - '0') * 1000 + (d1[1] - '0') * 100 + (d1[2] - '0') * 10 + (d1[3] - '0');
    ey = (d2[0] - '0') * 1000 + (d2[1] - '0') * 100 + (d2[2] - '0') * 10 + (d2[3] - '0');
    sm = (d1[4] - '0') * 10 + (d1[5] - '0');
    sd = (d1[6] - '0') * 10 + (d1[7] - '0');
    em = (d2[4] - '0') * 10 + (d2[5] - '0');
    ed = (d2[6] - '0') * 10 + (d2[7] - '0');
    for (i = sy; i <= ey; i++) {
        m = (sy % 10) * 10 + (sy / 10) % 10;
        d = ((sy / 100) % 10) * 10 + (sy / 1000);
        //对应的月份日期合理了才需要进一步判断这个时间是否在给定的时间区间内
        if (m > 0 && m < 13) {
            if ((i % 4 == 0 && i % 100 != 0) || i % 400 == 0) {
                month[2] = 29;
            }
            if (d > 0 && d <= month[m]) {
                if (i > sy && i < ey) {
                    sum++;
                }
                else if (i == sy && i == ey) {
                    if (hous() && qiane()) {
                        sum++;
                    }
                }
                else if (i == sy && i != ey) {
                    if (hous()) {
                        sum++;
                    }
                }
                else {
                    if (qiane()) {
                        sum++;
                    }
                }
            }
        }
        month[2] = 28;
    }
    cout << sum;
    return 0;
}

解法二(月日):

我们可以直接遍历所有的月日对应的回文日期,因为一年最多366天,遍历一次只需要366次,时间复杂度并不高,然后我们只需要判断这个日期是否在时间区间内即可,可以直接用整数,因为时间越往后,日期对应的整数一定越大,所以我们只需要比大小即可。

#include<iostream>
using namespace std;
int month[13] = { 0,31,29,31,30,31,30,31,31,30,31,30,31 };
int main(){
    ios::sync_with_stdio(false);
    cin.tie(0);cout.tie(0);
    int s,e,i,j,y,date,sum=0;
    cin>>s>>e;
    //从月份日期去找年份,看是否在区间范围内
    //其中0229对应的年份是9220是闰年,是合理的存在,所以我们直接把2月改成29天即可考虑到所有回文日期
    for(i=1;i<=12;i++){
        for(j=1;j<=month[i];j++){
            y=(j%10)*1000+(j/10)*100+(i%10)*10+i/10;
            date=y*10000+i*100+j;
            if(date>=s&&date<=e){
                sum++;
            }
        }
    }
    cout<<sum;
    return 0;
}

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值