【C++】每周一题——上周的

2024-02-22 08:36
文章标签 c++ 每周 上周

本文主要是介绍【C++】每周一题——上周的,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目

回文日期
在日常生活中,通过年、月、日这三个要素可以表示出一个唯一确定的日期。
牛牛习惯用 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 之间,有多少个日期是回文的。
【输入样例 1】
20110101 20111231
【输出样例 1】
1
样例 1 说明:
对于样例 1,符合条件的日期是 20111102
【输入样例 2】
20000101 20101231
【输出样例 2】
2
样例 2 说明:
对于样例 2,符合条件的日期是 20011002 和 20100102


题目分析

遇到长的题目,我们就先分析一下,理理思路。
输入两个日期,判断从第一个日期一直到第二个日期中,有多少日期是回文的。


解题

代码分析

先思考再做题的方法

  1. 我们不难想到,日期的格式为“年{4}月{2}日{2}”;而,6位的回文数即第一位=倒数第一位第二位=倒数第二位……我们其实可以直接遍历年,然后算出月和日,判断不符合真实的日期即可。
    ok,那就先写上框架开始。

    int main(){return 0;
    }
    

  2. 直接输入两个日期。

    #include<cstdio>	//scanf();
    int date1, date2;
    int main(){scanf("%d %d", &date1, &date2);return 0;
    }
    

  3. 从数据中提取出年,并遍历年。

    #include<cstdio>	//scanf()
    int date1, date2, year;
    int main(){scanf("%d %d", &date1, &date2);for(int year=date1/1000; year<=date2; year++){}return 0;
    }
    

  4. 算出准确日期。
    用到的方法很简单,数位分离反转数字。(不懂的点链接去看看)

    #include<cstdio>	//scanf()
    int date1, date2, tmp, month, day;
    int main(){scanf("%d %d", &date1, &date2);for(int year=date1/1000; year<=date2; year++){tmp=year;while(tmp>99){day*=10;day+=tmp%10;tmp/=10;}while(tmp>0){month*=10;month+=tmp%10;tmp/=10;}}return 0;
    }
    

  5. 判断当前遍历到的回文日期是否真实存在,如果在,计数器就自增1。
    一年12个月,这个好说,但是日就不好说了。不好说也没办法,用个switch,判断一下。

    #include<cstdio>	//scanf()
    int date1, date2, tmp, month, day, max, number;
    int main(){scanf("%d %d", &date1, &date2);for(int year=date1/1000; year<=date2; year++){tmp=year;while(tmp>99){day*=10;day+=tmp%10;tmp/=10;}while(tmp>0){month*=10;month+=tmp%10;tmp/=10;}if(month>=1 && month<=12){switch(month){case 1: case 3: case 5: case 7: case 8: case 10: case 12:max=31;break;case 4: case 6: case 9: case 11:max=30;break;default:if((year%100!=0 && year%4==0) || year%400==0){max=29;}else{max=28;}}if(day>=1 && day<=max){number++;}}}return 0;
    }
    

  6. 最后输出一下计数器的值就好了。

    #include<cstdio>    //scanf(), printf()
    int date1, date2, tmp, month, day, max, number;
    int main(){scanf("%d %d", &date1, &date2);for(int year=date1/1000; year<=date2; year++){tmp=year;while(tmp>99){day*=10;day+=tmp%10;tmp/=10;}while(tmp>0){month*=10;month+=tmp%10;tmp/=10;}if(month>=1 && month<=12){switch(month){case 1: case 3: case 5: case 7: case 8: case 10: case 12:max=31;break;case 4: case 6: case 9: case 11:max=30;break;default:if((year%100!=0 && year%4==0) || year%400==0){max=29;}else{max=28;}}if(day>=1 && day<=max){number++;}}}printf("%d", number);return 0;
    }
    


直接去做题的方法

没错,确实有不先肯思考的人;也不用多想,那个人就是我。这样循环次数多365倍,简直不要太慢。
不值得参考的代码放出来,别学我!

#include <iostream>
int daye, num, day[4], t;
int j(int y, int m){switch(m){case 1: case 3: case 5: case 7: case 8: case 10: case 12:return 31;case 4: case 6: case 9: case 11:return 30;default:if((y%100!=0 && y%4==0) || y%400==0){return 29;}else{return 28;}}
}
void d(){day[3]=day[0]%100;day[2]=day[0]/100%100;day[1]=day[0]/10000;
}
void p(){day[0]=day[1]*10000+day[2]*100+day[3];
}
int main(){scanf("%d %d", &day[0], &daye);for(d(); day[0]<=daye; day[3]++){if(day[3]>j(day[1], day[2])){day[3]-=j(day[1], day[2]);day[2]++;}if(day[2]>12){day[2]-=12;day[1]++;}p();t=0;for(int i=day[0]; i; i/=10){t*=10;t+=i%10;}if(t==day[0]){num++;}}printf("%d", num);return 0;
}

完整代码

  1. #include<cstdio>
    int date1, date2, tmp, month, day, max, number;
    int main(){scanf("%d %d", &date1, &date2);for(int year=date1/1000; year<=date2; year++){tmp=year;while(tmp>99){day*=10;day+=tmp%10;tmp/=10;}while(tmp>0){month*=10;month+=tmp%10;tmp/=10;}if(month>=1 && month<=12){switch(month){case 1: case 3: case 5: case 7: case 8: case 10: case 12:max=31;break;case 4: case 6: case 9: case 11:max=30;break;default:if((year%100!=0 && year%4==0) || year%400==0){max=29;}else{max=28;}}if(day>=1 && day<=max){number++;}}}printf("%d", number);return 0;
    }
    

  2. #include <iostream>
    int daye, num, day[4], t;
    int j(int y, int m){switch(m){case 1: case 3: case 5: case 7: case 8: case 10: case 12:return 31;case 4: case 6: case 9: case 11:return 30;default:if((y%100!=0 && y%4==0) || y%400==0){return 29;}else{return 28;}}
    }
    void d(){day[3]=day[0]%100;day[2]=day[0]/100%100;day[1]=day[0]/10000;
    }
    void p(){day[0]=day[1]*10000+day[2]*100+day[3];
    }
    int main(){scanf("%d\n%d", &day[0], &daye);for(d(); day[0]<=daye; day[3]++){if(day[3]>j(day[1], day[2])){day[3]-=j(day[1], day[2]);day[2]++;}if(day[2]>12){day[2]-=12;day[1]++;}p();t=0;for(int i=day[0]; i; i/=10){t*=10;t+=i%10;}if(t==day[0]){num++;}}printf("%d", num);return 0;
    }
    

尾声(鸡汤)

这个故事告诉我们做题做事前一定要先思考,不要贸然冲动,最后麻烦了自己可能也麻烦了别人。


这道题用到的算法是模拟算法,使用相同算法的题目还有这些,可以看看。
《【C++】每周一题——1024.2.21》
《【C++】高精度加减法》

这篇关于【C++】每周一题——上周的的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/734701

相关文章

c++ 类成员变量默认初始值的实现

《c++类成员变量默认初始值的实现》本文主要介绍了c++类成员变量默认初始值,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录C++类成员变量初始化c++类的变量的初始化在C++中,如果使用类成员变量时未给定其初始值,那么它将被

C++中NULL与nullptr的区别小结

《C++中NULL与nullptr的区别小结》本文介绍了C++编程中NULL与nullptr的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编... 目录C++98空值——NULLC++11空值——nullptr区别对比示例 C++98空值——NUL

C++ Log4cpp跨平台日志库的使用小结

《C++Log4cpp跨平台日志库的使用小结》Log4cpp是c++类库,本文详细介绍了C++日志库log4cpp的使用方法,及设置日志输出格式和优先级,具有一定的参考价值,感兴趣的可以了解一下... 目录一、介绍1. log4cpp的日志方式2.设置日志输出的格式3. 设置日志的输出优先级二、Window

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C