• 蓝桥杯-回文日期[Java]


    目录:

    学习目标:

    学习内容:

    学习时间:

    题目:

    题目描述:

    输入描述:

    输出描述:

    输入输出样例:

    示例 1:

    运行限制:

    题解:

    思路:


    学习目标:

    • 刷蓝桥杯题库日记

    学习内容:

    • 编号498
    • 题目回文日期
    • 难度困难

    学习时间:

    • 2023/11/4 17:00

    题目:

    题目描述:

    2020 年春节期间,有一个特殊的日期引起了大家的注意:2020 年 2 月 2 日。因为如果将这个日期按 “yyyymmdd” 的格式写成一个 8 位数是 20200202,恰好是一个回文数。我们称这样的日期是回文日期。

    有人表示 20200202 是 “千年一遇” 的特殊日子。对此小明很不认同,因为不到 2 年之后就是下一个回文日期:20211202 即 2021 年 12 月 2 日。

    也有人表示 20200202 并不仅仅是一个回文日期,还是一个 ABABBABA 型的回文日期。对此小明也不认同,因为大约 100 年后就能遇到下一个 ABABBABA 型的回文日期:21211212 即 2121 年 12 月 12 日。算不上 “千年一遇”,顶多算 “千年两遇”。

    给定一个 8 位数的日期,请你计算该日期之后下一个回文日期和下一个 ABABBABA 型的回文日期各是哪一天。

    输入描述:

    输入包含一个八位整数N,表示日期。

    对于所有评测用例,10000101 ≤ N ≤ 89991231 ,保证N是一个合法日期的 8 位数表示。

    输出描述:

    输出两行,每行 1 个八位数。第一行表示下一个回文日期,第二行表示下一个 ABABBABA 型的回文日期。

    输入输出样例:

    示例 1:

    输入:

    20200202

    输出:

    1. 20211202
    2. 21211212

    运行限制:

    • 最大运行时间:1s
    • 最大运行内存: 256M

    题解:

    1. import java.util.Scanner;
    2. // 1:无需package
    3. // 2: 类名必须Main, 不可修改
    4. public class Main {
    5. public static void main(String[] args) {
    6. Scanner scan = new Scanner(System.in);
    7. int start = scan.nextInt();
    8. int year = start / 10000;
    9. int goal = start % 10000;
    10. int nextyear = initnextyear(year,goal);
    11. int nextgoalyear = initnextgoalyear(year,goal);
    12. scan.close();
    13. System.out.println(nextyear);
    14. System.out.println(nextgoalyear);
    15. }
    16. public static int initnextyear(int year,int goal) {
    17. for(int i = year;i < 10000;i++){
    18. int m = reverse(i % 100);
    19. int d = reverse(i / 100);
    20. if(m > 12 || d > 31 || (!IsLeapYear(year) && d > 28 && m == 2) || (IsLeapYear(year) && d > 29 && m == 2) || m < 1 || d < 1)
    21. continue;
    22. else if((m*100+d > goal && i == year) || i > year)
    23. return i*10000 + m*100 + d;
    24. else
    25. continue;
    26. }
    27. return -1;
    28. }
    29. public static int initnextgoalyear(int year,int goal) {
    30. for(int i = year;i < 10000;i++){
    31. int m = reverse(i % 100);
    32. int d = reverse(i / 100);
    33. if(m != d)
    34. continue;
    35. else if(m > 12 || d > 31 || (!IsLeapYear(year) && d > 28 && m == 2) || (IsLeapYear(year) && d > 29 && m == 2) || m < 1 || d < 1)
    36. continue;
    37. else if((m*100+d > goal && i == year) || i > year)
    38. return i*10000 + m*100 + d;
    39. else
    40. continue;
    41. }
    42. return -1;
    43. }
    44. public static int reverse(int num) {
    45. int a = num / 10;
    46. int b = num % 10;
    47. return b*10 + a;
    48. }
    49. public static boolean IsLeapYear(int year) {
    50. if(year % 400 == 0 || (year % 4 == 0 && year % 100 != 0))
    51. return true;
    52. else
    53. return false;
    54. }
    55. }

    思路:

    • 有一个坑点,这个数的最大范围应该是99999999,而不是89991231。89991231为输入 可以得到90011009和90900909 只是说输入最大为89991231,没说输出最大为89991231。不然测试6会被卡住
    • 感觉自己写麻烦了,是因为自己最开始审错题了悲(,开始以为只要输入年份然后输出两个回文日期,后来才发现是输入一个日期。所以就缝缝补补了一下开头的start和goal。
    • 回归正题,主要思路还是判断日期的可行性。月份仅有1-12,日期仅有1-31,所以不在范围内的日期统统pass掉。
    • 考虑一下闰年,闰年的定义:整百年要被400整除,非整百年被4整除。闰年2月是29天,非闰年28天。
    • reverse函数用来针对年份进行反转得到月份和日期的。写这么麻烦就是因为最开始对题目的审题不仔细。
    • 用for循环找到正确的年份,即i循环到的值为年份。m和d对应的就是月份和日期。
    • 用输入值的年份作为i的起始点,如果i和year相等,用日期和goal进行比较大小,如果小于等于则continue,反之可直接返回。如果i大于year,则可直接返回。
  • 相关阅读:
    Unity 制作KinematicCharacterController
    华清远见(上海中心)22071
    【附源码】计算机毕业设计JAVA教学成果管理平台
    腾讯云4核8G服务器收费贵不贵?
    基于51单片机简易十字路口交通灯_5s全黄闪烁
    画面消失,但功能还在正常实现,窗口绘制出了问题
    编程参考- 重名符号的链接问题
    vue3(二)
    初阶数据结构:链表相关题目练习(补充)
    关于Qt5中QListWidgetItem不显示的问题
  • 原文地址:https://blog.csdn.net/Tanhauser/article/details/134228885