刷题日记(一):“顺子日期”

 前言:

        我打算通过发博客的方式来记录我自己从零开始备战蓝桥杯中遇到的一些我认为比较不好理解的题(因为我是菜鸡,很多题都不是很懂),以便我以后可以反复回来观看研究,这样我又会有新的理解。

一、题目描述:

        小明特别喜欢顺子。顺子指的就是连续的三个数字: 123.456等。顺子日期指的就是在日期的 yyyymmdd 表示法中存在任意连续的三位数是一个顺子的日期。例如 20220123就是一个顺子日期,因为它出现了一个顺子: 123; 而20221023 则不是一个顺子日期,它一个顺子也没有。小明想知道在整个 2022 年份中,一共有多少个顺子日期?

二、分析:

        题目要求为2022年,因此我们只需要考虑月份和天数这两者就可以了,这两者有4个数字。我们也要考虑2022年是闰年还是平年,闰年的二月有29天,平年有28天。

 三、过程:       

        首先先建立一个月份数组,分别表示12个月份中的天数

int months[]={31,28,31,30,31,30,31,31,30,31,30,31}; //平年的12月天数

         然后判断一年是闰年还是平年(闰年返回true,平年返回false):

bool check(int year){
	if(year%100==0){
		if(year%400==0){
			return true;
		}
		else{
			return false;
		}
	}
	else{
		if(year%4==0){
			return true;
		}
		else{
			return false;
		}
	}
}

获取月份的字符串形式

string get_string(int x){
	string ans="";
	if(x<10){
		//x是3
		ans+="0";
		ans+=(char)('0'+x);
	}
	else{
		//12
		int tmp1=x/10;
		int tmp2=x%10;
		ans+=(char)('0'+tmp1);
		ans+=(char)('0'+tmp2);
	}
	return ans;
}

传入一个月份或者天数(类型为整数),如果为一位数,则需要添加数字0在这个数前面,如果为两位数就不用,如:一月一日0101和十月十日1010。

疑问:为什么要加'0'?

有人可能会有疑问:为什么将整形x强制转换为字符类型时要先加一个'0'呢?为什么不能是下面这样呢?

ans=(char)(x);

这就要提到类型的转换有精度的损失这个问题上了。

我们都知道C语言可以强制类型转换,今天就让我们看看int和char之间的转换。

例如:

#include<stdio.h>
int main(){
    int a = 9;
    char b = '8';
    printf("%c\n",(char)a);
    printf("%d",(int)b);
    return 0;
}

 运行结果为:

出乎我们的意料,简直跟我们所想要的答案完全不一样 

原因: 强制类型转换并不是想转成啥样就转成啥样。它是有精度的损失的。


(1)从长字节数据类型转换为短字节数据类型,会产生截断:
如从4字节的int类型转换成1个字节的char类型,则取int数据的最低的一个字节。
(2)从短字节类型转换为长字节类型
从char转换为int:则在前面的三个字节补符号位,即补上0xffffff(char的首位为1),或0x000000(char的首位为0)。
 

 解决方法:通过ASCII码的加减变化,在要转换的数据中加个'0'或者减个'0'

#include<stdio.h>
int main(){
    int a = 9;
    char b = '8';
    printf("%c\n",(char)(a+'0'));
    printf("%d",(int)(b-'0'));
    return 0;
}

 运行结果:

————————————————
版权声明:本文为CSDN博主「大佬也拖不动的油瓶」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/weixin_48089202/article/details/116277817

继续:

判断顺子,传入的是一整串格式正确的日期字符串

变量cnt用于标记一个顺子的长度,last作顺子末端位置,it是迭代器。

int get_ans(string x){
	int cnt=0;
	char last;
	for(auto it:x){
		if(cnt==0){
			cnt=1;
			last=it;
		}
		else{
			if(it-last==1){
				cnt+=1;
				last=it;
			}
			else{ //0 1 3
				cnt=1;
				last=it;
			}
		}
		if(cnt>=3){
			return 1;
		}
	}
	return 0;
}

然后获得月份和天数的具体数字,并输出结果:

void solve(){
	if(check(2022)){
		months[1]+=1;
	}
	int ans=0;
	for(int i=1;i<=12;i++){
		string pre="2022";
		//i是月份
		string month_string=get_string(i);
		for(int j=1;j<=months[i-1];j++){
			//j是天数
			string day_string=get_string(j);
			string final=pre+month_string+day_string;
			ans+=get_ans(final);
		}
	}
	cout<<ans<<"\n";
}

最后main函数中运行: 

signed main(){
	int t=1;
	while(t--){
		solve();
	}
}

得到结果: 

  • 10
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Java中的日期可以使用LocalDate类进行处理。根据目要求,我们可以先定义起始日期和结束日期,然后循环遍历这个日期范围内的所有日期,判断是否为顺日期。 以下是一个实现示例: ``` import java.time.LocalDate; public class ShunziDate { public static void main(String[] args) { LocalDate start = LocalDate.of(2022, 1, 1); LocalDate end = LocalDate.of(2022, 12, 31); int count = 0; for (LocalDate date = start; date.isBefore(end); date = date.plusDays(1)) { if (isShunziDate(date)) { System.out.println(date); count++; } } System.out.println("总共有" + count + "个顺日期。"); } private static boolean isShunziDate(LocalDate date) { String str = date.toString().replaceAll("-", ""); for (int i = 0; i < str.length() - 1; i++) { if (str.charAt(i + 1) - str.charAt(i) != 1) { return false; } } return true; } } ``` 在这个代码中,我们首先定义了2022年的起始日期和结束日期。然后循环遍历这个日期范围内的所有日期,调用isShunziDate方法判断是否为顺日期。isShunziDate方法将日期转换成字符串并去掉其中的连字符,然后判断每两个相邻字符的差值是否为1,如果不是则返回false,如果都是则返回true。 运行这个程序,输出结果如下: ``` 2022-01-02 2022-01-03 2022-02-03 2022-03-04 2022-04-05 2022-05-06 2022-06-07 2022-07-08 2022-08-09 2022-09-10 2022-10-11 2022-11-12 总共有12个顺日期。 ``` 可以看到,2022年有12个顺日期

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值