用单片机制作最高精度时间系统,可移植到任何时钟程序中
可移植到任何数字时钟程序中,根据单片机型号以及晶振频率修改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秒
 }
}