|
用单片机制作最高精度时间系统,可移植到任何时钟程序中
可移植到任何数字时钟程序中,根据单片机型号以及晶振频率修改TH0和TL0的值。
智能日期时间累加,平闰年判断,当月天数判断功能,年计数值0~9999。
使用1秒的延迟函数,再加上时间判断计数函数的时间,误差非常大。
采用5毫秒定时器中断计数,中断触发后重置计数值并且5毫秒计数加1。
处理5毫秒计数以及时间判断计数函数,定时器仍在计数,准确触发高精度的5毫秒中断。
5毫秒计数加到200时,5毫秒计数清零并且触发时间计数函数,包含所有判断日期时间的指令在5毫
秒内完成,如5毫秒内无法完成请改成更长毫秒时间中断1次,避免出现中断重入等问题。
若发现bug或其他建议意见请及时在楼下回复。
下面是源码:
- #include "reg51.h"
- #include "intrins.h"
- sfr AUXR = 0x8E;
- sbit led=P3^2; //秒闪烁指示灯
- #define ui unsigned int
- #define uc unsigned char
- ui y; //定义变量年
- uc m,d,h,i,s,w; //定义变量月、日、时、分、秒、周
- uc ms5; //5毫秒定时器中断计数
- void InitTimer0(void){ //初始化5毫秒定时器,根据晶振频率修改TH0和TL0的值。
- TMOD = 0x01;
- TH0 = 0x28;
- TL0 = 0x00;
- EA = 1;
- ET0 = 1;
- TR0 = 1;
- }
- bit if_leap_year(ui y){ //平闰年检测,闰年返回1,平年返回0
- //闰年是4的倍数且不是100的倍数或者是400的倍数,否则是平年。
- if((y%4 == 0 && y%100 != 0) || y%400 == 0){
- return 1; //返回1表示闰年
- } else { //否则就是平年
- return 0; //返回0表示平年
- }
- }
- uc get_mon_day_nbr(ui y,uc m){ //通过年月获取该月有多少天
- if(m == 1 || m == 3 || m == 5 || m == 7 || m == 8 || m == 10 || m == 12) { //大
- 月则31天
- return 31;
- }
- if(m == 4 || m == 6 || m == 9 || m == 11){ //小月则30天
- return 30;
- }
- if(m == 2){ //2月
- if(if_leap_year(y) == 1){ //闰年29天
- return 29;
- } else { //平年28天
- return 28;
- }
- }
- return 0;
- }
- bit dt_is_ok(ui y,uc m,uc d,uc w,uc h,uc i,uc s){ //检测设置的日期时间格式是否有效 1有
- 效0无效
- if(w > 7 || w < 1) return 0; //星期大于7或小于1 返回0
- if(y > 9999 || m > 12 || m < 1 || d < 1) return 0; //年大于999 或 月大于12 或
- 月小于1 或日小于1 返回0
- if(d > get_mon_day_nbr(y,m)) return 0; //通过年月获取当月天数 大于该值 返回0
- if(h > 23 || i > 59 || s > 59) return 0; //时大于23 或 分大于59 或秒大于59 返回
- 0
- return 1; //检测有效 返回1
- }
- bit set_time(ui yy,uc mm,uc dd,uc ww,uc hh,uc ii,uc ss){ //设置日期时间
- if(dt_is_ok(yy,mm,dd,ww,hh,ii,ss) == 1){ //检查日期时间格式有效
- y=yy;m=mm;d=dd;w=ww; //设置年月日周
- h=hh;i=ii;s=ss; //设置时分秒
- return 1; //设置成功 返回1
- } else { //格式无效 返回0
- return 0;
- }
- }
- void add_time(){ //时间步进1秒
- s++; //秒加1
- if(s>=60){ //秒加到60
- s=0; //秒清零
- i++; //分加1
- if(i>=60){ //分加到60
- i=0; //分清零
- h++; //时加1
- if(h>=24){ //时加到24
- h=0; //时清零
- d++; //日加1
- w++; //星期加1
- if(w > 7){ //星期超过7
- w=1; //星期等于1
- }
- if(d > get_mon_day_nbr(y,m)){ //日超过当前年月的天数
- d=1; //日清1
- m++; //月加1
- if(m > 12){ //月加到12以上
- m=1; //月清1
- y++; //年加1
- if(y >= 10000){ //年加到10000
- y=0; //年清零
- }
- }
- }
- }
- }
- }
- }
- void main(){ //入口函数,上电复位后在此开始执行指令。
- P3=0xFF; //设置P3口全部为高电平
- set_time(2000,1,1,6,0,0,0); //设置日期时间
- AUXR |= 0x80; //关闭定时器0的12分频
- InitTimer0(); //初始化5毫秒定时器
- //死循环内可添加其他程序,只有中断触发后退出,中断返回后继续在原地运行程序。
- while(1);
- }
- void Timer0Interrupt(void) interrupt 1{ //5毫秒定时器中断
- //重置TH0与TL0的值,根据晶振频率修改TH0和TL0的值。
- TH0 = 0x28;
- TL0 = 0x00;
- ms5++; //5毫秒中断计数加1
- if(ms5%100 == 0)led=~led;
- //5毫秒中断计数加到200 200*5=1000毫秒 1秒触发1次
- if(ms5 >= 200){
- ms5=0; //5毫秒计数清零
- add_time(); //时间步进1秒
- }
-
- }
复制代码
|
评分
-
查看全部评分
|