Acwing---446.回文日期

1.题目

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

牛牛习惯用 8 位数字表示一个日期,其中,前 4 位代表年份,接下来 2 位代表月份,最后 2 位代表日期。

显然:一个日期只有一种表示方法,而两个不同的日期的表示方法不会相同。

牛牛认为,一个日期是回文的,当且仅当表示这个日期的 8 位数字是回文的。

现在,牛牛想知道:在他指定的两个日期之间(包含这两个日期本身),有多少个真实存在的日期是回文的。

一个 8 位数字是回文的,当且仅当对于所有的 i ( 1 ≤ i ≤ 8 ) i(1≤i≤8) i(1i8) 从左向右数的第 i i i 个数字和第 9 − i 9−i 9i个数字(即从

右向左数的第 i i i 个数字)是相同的。

例如:

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

输入格式
输入包括两行,每行包括一个 8 位数字。

第一行表示牛牛指定的起始日期 date1,第二行表示牛牛指定的终止日期 date2。保证 date1 和 date2 都是真实存在的日期,且年份部分一定为 4 位数字,且首位数字不为 0。

保证 date1 一定不晚于 date2。

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

输入样例:

20110101
20111231

输出样例:

1

2.基本思想

(枚举,模拟) O(10^4)
1.由于只有八位数,而且回文串左右对称,因此可以只枚举左半边,这样只需枚举 1000∼9999 个数,然后判断:

2.是否在范围内;

3.整个八位数构成的日期是否合法;

时间复杂度
一共枚举 104 个数,判断每个数是否合法的计算量是常数级别的,因此总计算量是 O(104)。
Tips
闰年需要满足下列两个条件的其中之一:
1.能被4整除但不能被100整除
2.能被400整除

3.代码实现

import java.util.Scanner;

public class _466回文日期 {
    static Scanner sc = new Scanner(System.in);
    static int[] months = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31};//月份表

    public static void main(String[] args) {
        int date1 = sc.nextInt();
        int date2 = sc.nextInt();
        int res = 0;
        for (int i = 1000; i <= 9999; i++) {
            int x = i, r = i;
            for (int j = 0; j < 4; j++) {// 1234 循环拼接4321  得到12344321
                x = x * 10 + r % 10;
                r /= 10;
            }
            //得到回文数 判断是否在给定范围 以及 判断日期是否合法
            if (date1 <= x && x <= date2 && check(x)) res++;
        }
        System.out.println(res);
    }

    //判断日期是否合法
    static boolean check(int date) {
        int year = date / 10000;
        int month = date / 100 % 100;
        int day = date % 100;
        //月份判定
        if (month == 0 || month > 12 || day == 0) return false;
        if (month != 2 && day > months[month]) return false;
        if (month == 2) {//2月
            boolean leap = (year % 400 == 0) || (year % 4 == 0 && year % 100 != 0);
            if (leap)//闰年
                if (day > 30) return false;
                else {//非闰年
                    if (day > 29) return false;
                }
        }
        return true;
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
好的,我可以帮你解决这个问题。 以下是一种解法: ```c #include <stdio.h> int is_leap(int year) { // 判断闰年 return (year % 4 == 0 && year % 100 != 0) || year % 400 == 0; } int is_palindrome(int num) { // 判断回文数 int temp = num, reversed = 0; while (temp > 0) { reversed = reversed * 10 + temp % 10; temp /= 10; } return reversed == num; } int main() { int year, month, day; scanf("%d %d %d", &year, &month, &day); int days_of_month[13] = {0, 31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31}; // 各月天数 if (is_leap(year)) { // 闰年 2月天数为 29 days_of_month[2] = 29; } int count = 0; for (int m = 1; m <= 12; m++) { // 枚举月份 if (m == month) { // 跳过当前月份 continue; } for (int d = 1; d <= days_of_month[m]; d++) { // 枚举日期 if (d == day) { // 跳过当前日期 continue; } int num = year * 10000 + m * 100 + d; // 构造日期数字 if (is_palindrome(num)) { // 判断是否回文日期 printf("%04d-%02d-%02d\n", year, m, d); count++; } } } if (count == 0) { // 没有回文日期 printf("INVALID\n"); } return 0; } ``` 该程序输入一个日期(年、月、日),输出该年中除了该日期之外的所有回文日期。 具体思路是: 1. 定义一个函数 `is_leap` 判断闰年,该函数输入一个年份,返回 1 表示该年为闰年,返回 0 表示该年为平年。 2. 定义一个函数 `is_palindrome` 判断回文数,该函数输入一个整数,返回 1 表示该整数为回文数,返回 0 表示该整数不是回文数。 3. 在 `main` 函数中读入一个日期(年、月、日),并根据该年是否为闰年确定 2 月的天数。 4. 枚举除了该日期之外的所有日期,计算其对应的数字,并调用 `is_palindrome` 函数判断是否为回文数。 5. 如果存在回文日期,则输出该日期;否则输出 `INVALID`。 希望能帮到你!
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

amant 柒少

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值