🏆本文收录于《CSDN问答解惑-专业版》专栏,主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
问题描述
stm32按键设置闹钟数进退位不正常。软件是keil5,板子是STM32F407ZGTx。程序想实现LCD屏实时显示时间,并有闹钟功能,按键可以更改设置时间和闹钟。
写的完整代码如下:
#include "stm32f4xx.h"
#include "system.h"
#include "Systick.h"
#include "LED.h"
#include "BEEP.h"
#include "Key.h"
#include "EXTI.h"
#include "TIME.h"
#include "PWM.h"
#include "Usart.h"
#include "stdio.h"
#include "IWDG.h"
#include "WWDG.h"
#include "tftlcd.h"
#include "Draw.h"
#include "rtc.h"/*******************************************************************************
* 函 数 名 : main
* 函数功能 : 主函数
* 输 入 : 无
* 输 出 : 无
*******************************************************************************/
u8 GetMaxDay(u16 year, u8 month) //每月最大天数
{u8 day;switch (month) {case 1: case 3: case 5: case 7: case 8: case 10: case 12:day = 31;break;case 4: case 6: case 9: case 11:day = 30;break;case 2:// 判断是否是闰年if ((year % 4 == 0 && year % 100 != 0) || (year % 400 == 0)) {day = 29;} else {day = 28;}break;default:day = 31;break;}return day;
}int main()
{ unsigned char time_str[10];unsigned char date_str[25];unsigned char clock_str[25];//按键状态定义u8 key = 0;u8 k0=0,k1=0;//初始化SysTick_Init(168);//时钟频率为168 MHzNVIC_PriorityGroupConfig(NVIC_PriorityGroup_2);//配置NVIC的优先级分组,设置为优先级组2USART1_Init(115200);//波特率为115200LED_Init();//初始化LED模块TFTLCD_Init();//初始化TFT LCD显示屏KEY_Init();//初始化按键模块BEEP_Init();//初始化蜂鸣器模块RTC_Config();//配置实时时钟(RTC)FRONT_COLOR=BLACK;//初始化结构体变量RTC_DateStruct.RTC_Year = 24; // 假设年份为2024年RTC_DateStruct.RTC_Month = 6; // 假设月份为6月RTC_DateStruct.RTC_Date = 18; // 假设日期为18日RTC_DateStruct.RTC_WeekDay = 2; // 假设星期为周二RTC_TimeStruct.RTC_Hours = 11; // 假设小时为11RTC_TimeStruct.RTC_Minutes = 10;//假设分钟为10RTC_TimeStruct.RTC_Seconds = 0; // 假设秒为0RTC_AlarmStruct.RTC_AlarmDateWeekDay = 2; // 假设闹钟日期为周二RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours = 11; // 假设闹钟小时为11RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes = 11; // 假设闹钟分钟为11RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds = 0; // 假设闹钟秒为0RTC_Set_WakeUp(RTC_WakeUpClock_CK_SPRE_16bits,0);//配置WAKE UP中断,1秒钟中断一次//RTC_Set_AlarmA(3,16,31,50); //使用闹钟A,示例周3,16点31分50秒响RTC_Set_AlarmA(2,11,11,00); //周2 11点11分00秒 蜂鸣器响while(1){//按键改变key=Key_Scan(0);if(key == KEY0_PRESS){ k0++;printf("按键k0依次设置顺序年、月、日、星期、时、分、秒、闹钟星期、闹钟时、闹钟分、闹钟秒,key_up加1,key_1减1\r\n");if(k0 == 1){printf("进入年份设置\r\n");k1 = 1;}if(k0 == 2){printf("进入月份设置\r\n");k1 = 2;}if(k0 == 3){printf("进入日设置\r\n");k1 = 3;}if(k0 == 4){printf("进入星期设置\r\n");k1 = 4;}if(k0 == 5){printf("进入时设置\r\n");k1 = 5;}if(k0 == 6){printf("进入分设置\r\n");k1 = 6;}if(k0 == 7){printf("进入秒设置\r\n");k1 = 7;}if(k0 == 8){printf("进入闹钟的星期设置\r\n");k1 = 8;}if(k0 == 9){printf("进入闹钟的时设置\r\n");k1 = 9;}if(k0 == 10){printf("进入闹钟的分设置\r\n");k1 = 10;}if(k0 == 11){printf("进入闹钟的秒设置\r\n");k1 = 11;}}//改变实时时钟的年份if((key == KEY_UP_PRESS) && (k1 == 1)){printf("进入年份加1判断\r\n");RTC_DateStruct.RTC_Year++;RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}if((key == KEY1_PRESS) && (k1 == 1)){printf("进入年份减1判断\r\n");RTC_DateStruct.RTC_Year--;RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}//改变实时时钟的月份if((key == KEY_UP_PRESS) && (k1 == 2)){printf("进入月份加1判断\r\n");RTC_DateStruct.RTC_Month = (RTC_DateStruct.RTC_Month % 12) + 1; // 增加月份循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}if((key == KEY1_PRESS) && (k1 == 2)){printf("进入月份减1判断\r\n");RTC_DateStruct.RTC_Month = (RTC_DateStruct.RTC_Month + 10) % 12 + 1; // 减少月份循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}//改变实时时钟的日if((key == KEY_UP_PRESS) && (k1 == 3)){printf("进入日加1判断\r\n");u8 max_day = GetMaxDay(RTC_DateStruct.RTC_Year, RTC_DateStruct.RTC_Month);RTC_DateStruct.RTC_Date = (RTC_DateStruct.RTC_Date % max_day) + 1; // 增加日期并循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}if((key == KEY1_PRESS) && (k1 == 3)){printf("进入日减1判断\r\n");u8 max_day = GetMaxDay(RTC_DateStruct.RTC_Year, RTC_DateStruct.RTC_Month);RTC_DateStruct.RTC_Date = (RTC_DateStruct.RTC_Date + max_day - 2) % max_day + 1; // 减少日期并循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}//改变实时时钟星期if((key == KEY_UP_PRESS) && (k1 == 4)){printf("进入星期加1判断\r\n");RTC_DateStruct.RTC_WeekDay = (RTC_DateStruct.RTC_WeekDay % 7) + 1; // 增加星期并循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}if((key == KEY1_PRESS) && (k1 == 4)){printf("进入星期减1判断\r\n");RTC_DateStruct.RTC_WeekDay = (RTC_DateStruct.RTC_WeekDay + 5) % 7 + 1; // 减少星期并循环RTC_Set_Date(RTC_DateStruct.RTC_Year , RTC_DateStruct.RTC_Month , RTC_DateStruct.RTC_Date , RTC_DateStruct.RTC_WeekDay);}//改变实时时钟的时if((key == KEY_UP_PRESS) && (k1 == 5)){printf("进入时加1判断\r\n");RTC_TimeStruct.RTC_Hours = (RTC_TimeStruct.RTC_Hours + 1) % 24; // 增加小时并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}if((key == KEY1_PRESS) && (k1 == 5)){printf("进入时减1判断\r\n");RTC_TimeStruct.RTC_Hours = (RTC_TimeStruct.RTC_Hours + 23) % 24; // 减少小时并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}//改变实时时钟的分if((key == KEY_UP_PRESS) && (k1 == 6)){printf("进入分加1判断\r\n");RTC_TimeStruct.RTC_Minutes = (RTC_TimeStruct.RTC_Minutes + 1) % 60; // 增加分钟并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}if((key == KEY1_PRESS) && (k1 == 6)){printf("进入分减1判断\r\n");RTC_TimeStruct.RTC_Minutes = (RTC_TimeStruct.RTC_Minutes + 59) % 60; // 减少分钟并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}//改变实时时钟的秒if((key == KEY_UP_PRESS) && (k1 == 7)){printf("进入秒加1判断\r\n");RTC_TimeStruct.RTC_Seconds = (RTC_TimeStruct.RTC_Seconds + 1) % 60; // 增加秒并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}if((key == KEY1_PRESS) && (k1 == 7)){RTC_TimeStruct.RTC_Seconds = (RTC_TimeStruct.RTC_Seconds + 59) % 60; // 减少秒并循环RTC_Set_Time(RTC_TimeStruct.RTC_Hours,RTC_TimeStruct.RTC_Minutes,RTC_TimeStruct.RTC_Seconds,RTC_H12_AM);}//改变闹钟的星期if((key == KEY_UP_PRESS) && (k1 == 8)){printf("进入闹钟星期加1判断\r\n");RTC_AlarmStruct.RTC_AlarmDateWeekDay = (RTC_AlarmStruct.RTC_AlarmDateWeekDay % 7) + 1; // 增加星期并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}if((key == KEY1_PRESS) && (k1 == 8)){printf("进入闹钟星期减1判断\r\n");RTC_AlarmStruct.RTC_AlarmDateWeekDay = (RTC_AlarmStruct.RTC_AlarmDateWeekDay + 5) % 7 + 1; // 减少星期并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}//改变闹钟的时if((key == KEY_UP_PRESS) && (k1 == 9)){printf("进入闹钟的时加1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours + 1) % 24; // 增加小时并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}if((key == KEY1_PRESS) && (k1 == 9)){printf("进入闹钟的时减1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours + 23) % 24; // 减少小时并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}//改变闹钟的分if((key == KEY_UP_PRESS) && (k1 == 10)){printf("进入闹钟的分加1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes + 1) % 60; // 增加分钟并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}if((key == KEY1_PRESS) && (k1 == 10)){printf("进入闹钟的分减1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes + 59) % 60; // 减少分钟并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}//改变闹钟的秒if((key == KEY_UP_PRESS) && (k1 == 11)){printf("进入闹钟的秒加1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds + 1) % 60; // 增加秒并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}if((key == KEY1_PRESS) && (k1 == 11)){printf("进入闹钟的秒减1判断\r\n");RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds = (RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds + 59) % 60; // 减少秒并循环RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay,RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours,RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes,RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);}RTC_GetTime(RTC_Format_BIN,&RTC_TimeStruct);RTC_GetDate(RTC_Format_BIN,&RTC_DateStruct);RTC_GetAlarm(RTC_Format_BIN , RTC_Alarm_A , &RTC_AlarmStruct);//LCD_ShowString(10,10,tftlcd_data.width,tftlcd_data.height,12,"hello");
//snprintf(str, sizeof(str), "%d", num); //整数变字符串
//LCD_ShowString(10,10,tftlcd_data.width,tftlcd_data.height,12,str);sprintf(date_str, "20%02d-%02d-%02d week:%d", RTC_DateStruct.RTC_Year, RTC_DateStruct.RTC_Month, RTC_DateStruct.RTC_Date, RTC_DateStruct.RTC_WeekDay); // 整形转字符串sprintf(time_str, "%02d:%02d:%02d", RTC_TimeStruct.RTC_Hours, RTC_TimeStruct.RTC_Minutes, RTC_TimeStruct.RTC_Seconds); // 整形转字符串sprintf(clock_str, "week:%d %d:%d:%d", RTC_AlarmStruct.RTC_AlarmDateWeekDay, RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours, RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes, RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds); // 整形转字符串LCD_ShowString(10, 30, tftlcd_data.width, tftlcd_data.height, 24, date_str); // 实时时间LCD显示LCD_ShowString(10, 60, tftlcd_data.width, tftlcd_data.height, 24, time_str); // 实时时间LCD显示LCD_ShowString(10, 90, tftlcd_data.width, tftlcd_data.height, 24, clock_str); // 实时时间LCD显示}
}
烧录后发现,按键更改闹钟的秒数时,加减实现进位、退位有问题。
问题展示如图可见,第三行最右就是闹钟秒数。在闹钟秒为59时摁按键增加,没有跳转到00而是09,之后一直增加可到99,99再增加又跳到了10,10之后增加就是正常进位了;若在10时摁按键减少,就是99、89、79…到09这样子减少,继续减少又是59,59之后减少是正常的。按键调整其他,例如年月日星期什么的都是正常的,只有闹钟秒数这里有问题。
如上问题有来自我自身项目开发,有的收集网站,有的来自读者,如有侵权,立马删除。
解决方案
如下是上述问题的解决方案,仅供参考:
根据您的描述和代码,问题出现在处理闹钟秒数的进位和退位逻辑上。在您的代码中,对秒数的加1和减1操作使用了简单的模运算,但在处理边界情况(即59到60和0到-1)时,这种模运算会导致不正确的行为。
问题分析
-
加1问题:当秒数为59时,
(RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds + 1) % 60
的结果应该是0,但您的代码中是09
,这可能是因为在显示时没有考虑到秒数从59到0的转换。 -
减1问题:当秒数为10时,
(RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds + 59) % 60
的结果应该是-49,模60后变成11,这显然不正确。正确的逻辑应该是当秒数为0时,减1应该变成59。
解决方案
对于加1和减1的操作,我们需要重新设计逻辑,确保在边界情况下能够正确处理。
改进的加1逻辑
if((key == KEY_UP_PRESS) && (k1 == 11))
{RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds++;if(RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds > 59){RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds = 0;}RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay, RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours, RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes, RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);
}
改进的减1逻辑
if((key == KEY1_PRESS) && (k1 == 11))
{if(RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds == 0){RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds = 59;}else{RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds--;}RTC_Set_AlarmA(RTC_AlarmStruct.RTC_AlarmDateWeekDay, RTC_AlarmStruct.RTC_AlarmTime.RTC_Hours, RTC_AlarmStruct.RTC_AlarmTime.RTC_Minutes, RTC_AlarmStruct.RTC_AlarmTime.RTC_Seconds);
}
其他注意事项
- 确保在修改时间或闹钟设置后,调用
RTC_Set_AlarmA()
函数时传入的参数是正确的。 - 检查LCD显示函数
LCD_ShowString
是否正确处理了时间格式的显示,特别是在进位和退位后。 - 如果可能,添加一些调试信息,打印出在按键操作前后的秒数,以便于跟踪问题。
通过上述修改,应该能够解决您遇到的进位和退位不正常的问题。如果问题仍然存在,请检查其他可能影响显示和设置的部分代码。
希望如上措施及解决方案能够帮到有需要的你。
PS:如若遇到采纳如下方案还是未解决的同学,希望不要抱怨&&急躁,毕竟影响因素众多,我写出来也是希望能够尽最大努力帮助到同类似问题的小伙伴,即把你未解决或者产生新Bug黏贴在评论区,我们大家一起来努力,一起帮你看看,可以不咯。
若有对当前Bug有与如下提供的方法不一致,有个不情之请,希望你能把你的新思路或新方法分享到评论区,一起学习,目的就是帮助更多所需要的同学,正所谓「赠人玫瑰,手留余香」。
☀️写在最后
ok,以上就是我这期的Bug修复内容啦,如果还想查找更多解决方案,你可以看看我专门收集Bug及提供解决方案的专栏《CSDN问答解惑-专业版》,都是实战中碰到的Bug,希望对你有所帮助。到此,咱们下期拜拜。
码字不易,如果这篇文章对你有所帮助,帮忙给bug菌来个一键三连(关注、点赞、收藏) ,您的支持就是我坚持写作分享知识点传播技术的最大动力。
同时也推荐大家关注我的硬核公众号:「猿圈奇妙屋」 ;以第一手学习bug菌的首发干货,不仅能学习更多技术硬货,还可白嫖最新BAT大厂面试真题、4000G Pdf技术书籍、万份简历/PPT模板、技术文章Markdown文档等海量资料,你想要的我都有!
📣关于我
我是bug菌,CSDN | 掘金 | InfoQ | 51CTO | 华为云 | 阿里云 | 腾讯云 等社区博客专家,C站博客之星Top30,华为云2023年度十佳博主,掘金多年度人气作者Top40,51CTO年度博主Top12,掘金/InfoQ/51CTO等社区优质创作者;全网粉丝合计 20w+;硬核微信公众号「猿圈奇妙屋」,欢迎你的加入!免费白嫖最新BAT互联网公司面试真题、4000G PDF电子书籍、简历模板等海量资料,你想要的我都有,关键是你不来拿。