蓝桥杯103 日期问题

题目链接:https://www.lanqiao.cn/problems/103/learning/

前置知识

输入解析

要会什么?

会用这一句把 AA/BB/CC 读进来:

cpp 复制代码
int a,b,c;
scanf("%d/%d/%d", &a, &b, &c);

要记住什么?

"%d/%d/%d" 的意思是:

先读一个整数,接着必须遇到 / ,再读一个整数,再遇到 / ,再读一个整数

题目格式固定,所以不用担心空格、换行、乱格式

读进来的 a,b,c 都是 0~99 的两位数(比如 02 会读成整数 2)

两位年份到四位年份映射

要会什么?

把两位数的 yy 变成四位数年份:

cpp 复制代码
int fullYear(int yy) { 
    return (yy <= 59) ? (2000 + yy) : (1900 + yy);
}

要记住什么?

题目限定最终年份只能在 1960~2059,所以映射只能是:

00~59 → 2000~2059

60~99 → 1960~1999

条件运算符 ?: 的读法:

条件 ? A : B

条件成立返回 A,否则返回 B

闰年判断

要会什么?

写一个函数判断 y 是否为闰年:

cpp 复制代码
bool isLeap(int y) {
    return (y % 400 == 0) || (y % 4 == 0 && y % 100 != 0);
}

要记住什么?

能被 400 整除是闰年

能被 4 整除且不能被 100 整除也是闰年

其他是平年

运算符优先级:&& 比 || 高,但写括号最稳

月份天数表 + 日期合法性判断

要会什么?

用数组存每月天数,并写 valid(y,m,d) 判断是否合法:

cpp 复制代码
static int md[13] = {0,31,28,31,30,31,30,31,31,30,31,30,31};

bool valid(int y, int m, int d) {
    if (y < 1960 || y > 2059) return false;
    if (m < 1 || m > 12) return false;
    int days = md[m] + (m == 2 && isLeap(y)); // 闰年2月+1
    return d >= 1 && d <= days;
}

要记住什么?

md[0]=0 只是占位,让月份 1~12 直接当下标好用。

days = md[m] + (m==2 && isLeap(y))

这里 (m==2 && isLeap(y)) 结果是 true/false,在 C/C++ 里会当 1/0 用。

三种日期格式枚举思想

要会什么?

输入 a/b/c 可能代表三种解释方式,所以只需要枚举这三种:

yy/mm/dd → (fullYear(a), b, c)

mm/dd/yy → (fullYear(c), a, b)

dd/mm/yy → (fullYear(c), b, a)

要记住什么?

这题不是"枚举所有天",而是"枚举所有解释方式",最多 3 个候选,极快。

每个候选都用 valid() 筛一下,合法的才收集。

排序 + 去重

要会什么

把收集到的候选日期排序,然后去掉重复项:

sort(ans.begin(), ans.end());

ans.erase(unique(ans.begin(), ans.end()), ans.end());

要记住什么

不用死记位置,知道 unique 会标记重复区起点,erase 删掉这个起点后的所有元素就够了。

容器与数据结构表达

要会什么?

用一个能自动按年-月-日排序的结构存答案:

vector<pair<int, pair<int,int>>> ans; // (y, (m, d))

插入时可以这样:

ans.push_back({y, {m, d}});

要记住什么?

pair 默认比较规则是字典序:

先比 first(年)

年相同再比 second.first(月)

月相同再比 second.second(日)

所以你 sort(ans.begin(), ans.end()) 就会自动按年月日排好

格式化输出

要会什么?

输出成 yyyy-mm-dd,月日补 0:

printf("%d-%02d-%02d\n", y, m, d);

要记住什么?

%02d:宽度 2,不够左侧补 0

3 会输出成 03

完整代码

cpp 复制代码
#include <bits/stdc++.h>
using namespace std;

int fullYear(int yy) { // 00-59 -> 2000-2059, 60-99 -> 1960-1999
    return (yy <= 59) ? (2000 + yy) : (1900 + yy);
}

bool isLeap(int y) {
    return (y % 400 == 0) || (y % 4 == 0 && y % 100 != 0);
}

bool valid(int y, int m, int d) {
    if (y < 1960 || y > 2059) return false;
    if (m < 1 || m > 12) return false;
    static int md[13] = {0,31,28,31,30,31,30,31,31,30,31,30,31};
    int days = md[m] + (m == 2 && isLeap(y));
    return d >= 1 && d <= days;
}

int main() {
    int a, b, c;
    scanf("%d/%d/%d", &a, &b, &c);

    vector<pair<int, pair<int,int> > > ans; // (y, (m, d))

    int y1 = fullYear(a), m1 = b, d1 = c;   // yy/mm/dd
    int y2 = fullYear(c), m2 = a, d2 = b;   // mm/dd/yy
    int y3 = fullYear(c), m3 = b, d3 = a;   // dd/mm/yy

    if (valid(y1,m1,d1)) ans.push_back(make_pair(y1, make_pair(m1,d1)));
    if (valid(y2,m2,d2)) ans.push_back(make_pair(y2, make_pair(m2,d2)));
    if (valid(y3,m3,d3)) ans.push_back(make_pair(y3, make_pair(m3,d3)));

    sort(ans.begin(), ans.end());
    ans.erase(unique(ans.begin(), ans.end()), ans.end());

    for (int i = 0; i < ans.size(); ++i) {
        int y = ans[i].first;
        int m = ans[i].second.first;
        int d = ans[i].second.second;
        printf("%d-%02d-%02d\n", y, m, d);
    }
    return 0;
}
相关推荐
孞㐑¥10 分钟前
算法——BFS
开发语言·c++·经验分享·笔记·算法
MZ_ZXD0012 小时前
springboot旅游信息管理系统-计算机毕业设计源码21675
java·c++·vue.js·spring boot·python·django·php
A星空1233 小时前
一、Linux嵌入式的I2C驱动开发
linux·c++·驱动开发·i2c
凡人叶枫3 小时前
C++中智能指针详解(Linux实战版)| 彻底解决内存泄漏,新手也能吃透
java·linux·c语言·开发语言·c++·嵌入式开发
会叫的恐龙3 小时前
C++ 核心知识点汇总(第六日)(字符串)
c++·算法·字符串
小糯米6013 小时前
C++顺序表和vector
开发语言·c++·算法
独望漫天星辰4 小时前
C++ 多态深度解析:从语法规则到底层实现(附实战验证代码)
开发语言·c++
王老师青少年编程4 小时前
2024年信奥赛C++提高组csp-s初赛真题及答案解析(阅读程序第3题)
c++·题解·真题·csp·信奥赛·csp-s·提高组
凡人叶枫5 小时前
C++中输入、输出和文件操作详解(Linux实战版)| 从基础到项目落地,避坑指南
linux·服务器·c语言·开发语言·c++
CSDN_RTKLIB5 小时前
使用三方库头文件未使用导出符号情景
c++