【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++库方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录方法一:使用P/Invoke1. 导出C++函数2. 定义P/Invoke签名3. 调用C++函数方法二:使用C++/CLI作为桥接1. 创建C++/CL

C++如何通过Qt反射机制实现数据类序列化

《C++如何通过Qt反射机制实现数据类序列化》在C++工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作,所以本文就来聊聊C++如何通过Qt反射机制实现数据类序列化吧... 目录设计预期设计思路代码实现使用方法在 C++ 工程中经常需要使用数据类,并对数据类进行存储、打印、调试等操作。由于数据类

Linux下如何使用C++获取硬件信息

《Linux下如何使用C++获取硬件信息》这篇文章主要为大家详细介绍了如何使用C++实现获取CPU,主板,磁盘,BIOS信息等硬件信息,文中的示例代码讲解详细,感兴趣的小伙伴可以了解下... 目录方法获取CPU信息:读取"/proc/cpuinfo"文件获取磁盘信息:读取"/proc/diskstats"文

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

C++中初始化二维数组的几种常见方法

《C++中初始化二维数组的几种常见方法》本文详细介绍了在C++中初始化二维数组的不同方式,包括静态初始化、循环、全部为零、部分初始化、std::array和std::vector,以及std::vec... 目录1. 静态初始化2. 使用循环初始化3. 全部初始化为零4. 部分初始化5. 使用 std::a

C++ vector的常见用法超详细讲解

《C++vector的常见用法超详细讲解》:本文主要介绍C++vector的常见用法,包括C++中vector容器的定义、初始化方法、访问元素、常用函数及其时间复杂度,通过代码介绍的非常详细,... 目录1、vector的定义2、vector常用初始化方法1、使编程用花括号直接赋值2、使用圆括号赋值3、ve

如何高效移除C++关联容器中的元素

《如何高效移除C++关联容器中的元素》关联容器和顺序容器有着很大不同,关联容器中的元素是按照关键字来保存和访问的,而顺序容器中的元素是按它们在容器中的位置来顺序保存和访问的,本文介绍了如何高效移除C+... 目录一、简介二、移除给定位置的元素三、移除与特定键值等价的元素四、移除满足特android定条件的元

Python获取C++中返回的char*字段的两种思路

《Python获取C++中返回的char*字段的两种思路》有时候需要获取C++函数中返回来的不定长的char*字符串,本文小编为大家找到了两种解决问题的思路,感兴趣的小伙伴可以跟随小编一起学习一下... 有时候需要获取C++函数中返回来的不定长的char*字符串,目前我找到两种解决问题的思路,具体实现如下:

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

Java调用C++动态库超详细步骤讲解(附源码)

《Java调用C++动态库超详细步骤讲解(附源码)》C语言因其高效和接近硬件的特性,时常会被用在性能要求较高或者需要直接操作硬件的场合,:本文主要介绍Java调用C++动态库的相关资料,文中通过代... 目录一、直接调用C++库第一步:动态库生成(vs2017+qt5.12.10)第二步:Java调用C++