单片机论坛

 找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 1332|回复: 21
打印 上一主题 下一主题
收起左侧

单片机时钟24小时慢了21.5秒 如何修改初值校正呢?

  [复制链接]
跳转到指定楼层
楼主
单片机时钟24小时慢了21.5秒 如何修改初值呢

TH TL怎么计算减少多少呢

请帮助哈!!
分享到:  QQ好友和群QQ好友和群 QQ空间QQ空间 腾讯微博腾讯微博 腾讯朋友腾讯朋友
收藏收藏1 转播转播 分享分享 分享淘帖 顶1 踩
回复

使用道具 举报

来自 2#
造梦Sir 发表于 2018-9-4 08:37 | 只看该作者
听你的意思是这个时钟是基于单片机定时器走的,如果真的是这样走了24小时误差这么多不足为奇,如果你要求精度很高的话,就不要采用定时器,你可以用一些专门的实时时钟芯片,类似于DS1302这些。

评分

参与人数 1黑币 +50 收起 理由
admin + 50 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

来自 3#
cjm82 发表于 2018-9-5 10:11 | 只看该作者
24小时误差21.5秒,楼主用的应该是片上RC时钟,片上RC时钟存在0.1%-0.5%的误差,只适合于对计时要求不高且误差不会累加的场合,如利用中断实现对某一继电器实现延时功能,当信号保持时间超过10秒,继电器吸合,保持时间低于10秒则无动作.因为每次信号触发都会重置10秒的计时器,所以误差不会累加,而0.5%的误差对于10秒也可忽略不计,故用片上RC不会有太大问题,但用在计时时间较长,或者每次计时是基于上一次计时的基础上的场合,就会有明显的误差问题,这种场合建议使用晶振.

评分

参与人数 1黑币 +50 收起 理由
admin + 50 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

来自 4#
dsadsada12313 发表于 2018-9-27 11:26 | 只看该作者
单片机时钟可分为2种:一是应用时钟芯片;二是用机内时钟。应用时钟芯片肯定比机内时钟精准,它取决于时钟芯片的晶振频率是否精准。应用机内时钟一般是学生实验,让学生学习定时器中断、计时计次,无实用意义。

评分

参与人数 1黑币 +30 收起 理由
admin + 30 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

来自 5#
Y_G_G 发表于 2018-10-12 23:10 | 只看该作者
专业做表30年:
不要用单片机的时钟来做时钟源,这是不会准的,不管你怎么校正MHz级别的晶振用在秒上,对温度是比较敏感的,冬天和夏天都不一样的频率的,你想时钟准确,三个实用的方法:
1,如果你的石英精度校正机,可以用DS1302,不要看别人说的匹配多少的电容,每个器件都不会是一样的,校正好了,保你三年之内不用管.
2.用类似于DS3231这类已经校正好的时钟IC,精度是肯定可以的,价格会高一些
3.使用有源晶振,有了温度补偿精度才能保证,价格也会高一些
至于GPS,不怎么实用,至于WIFI有点难.

评分

参与人数 1黑币 +40 收起 理由
admin + 40 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

6#
wulin 发表于 2018-9-4 06:20 | 只看该作者
如果不使用专用时钟芯片而是使用定时器做时基,单片机时钟的精准度取决于晶振的精准度和编程技巧。可以使日误差控制在1秒。给你一个示例参考,基于TX-1C实验板。
//K1键调整选择,K2键+,长按连+,K3键-,长按连-,K4键闹钟设置
//定时器初始化程序根据晶振频率选择
//主程序循环一次必须小于100us,否则要更改定时器周期
//用计数法代替软件延时,提高走时精度
//数码管采用分时动态显示,约2ms显示1位
#include <reg51.h>
#define uint unsigned int
#define uchar unsigned char
#define key_S 300                                                //宏定义短按(约20ms)
#define key_L 6000                                        //宏定义长按(约2/3s)
#define key_M 3000                                        //宏定义长按(约1/3s)
sbit K1 = P3^4;                                                //调整选择/退出
sbit K2 = P3^5;                                                //++,长按连+
sbit K3 = P3^6;                                                //--,长按连-
sbit K4 = P3^7;                                                //调整时间
sbit dula=P2^6;                                                //段选
sbit wela=P2^7;                                                //位选
sbit Buzzer=P2^3;                                                //蜂鸣器
uchar code table[]={                                        //0~F数组
        0x3f,0x06,0x5b,0x4f,
        0x66,0x6d,0x7d,0x07,
        0x7f,0x6f,0x77,0x7c,
        0x39,0x5e,0x79,0x71};
uint Cnt100us;                                                        //定义100微秒变量
uchar hour=12,min=0,sec=0;                        //定义时、分、秒变量
uchar hour4=0,min4=0;                                //定义闹钟时、分变量
uchar  Choice=0,Choice4=0;                        //调整时间、闹钟选择变量
bit Buzzer_sign;                                                //蜂鸣器闹钟标志
bit Twinkle,second=1;                                //闪烁标志、秒标志
/**************************************
        定时器0初始化程序 100微秒@12.000MHz
**************************************/
/*
void Timer0Init()                                        //100微秒@12.000MHz
{
        TMOD = 0x02;                                        //设置自动重载模式
        TL0 = 0x9C;                                                //设置定时初值
        TH0 = 0x9C;                                                //设置定时重载值
        TF0 = 0;                                                //清除TF0标志
        TR0 = 1;                                                //定时器0开始计时
}
*/
/***************************************
        定时器0/1初始化程序 100微秒@11.0592MHz
***************************************/
void Timer_Init()                                        //100微秒@11.0592MHz
{
        TMOD = 0x22;                                        //设置自动重载模式
        TL0 = 0xA4;                                                //设置定时初值
        TH0 = 0xA4;                                                //设置定时重载值
        TL1 = 0x1A;                                        //设置定时初值250微秒@11.0592MHz
        TH1 = 0x1A;                                        //设置定时重载值
        TF0 = 0;                                                //清除TF0标志
        TR0 = 1;                                                //定时器0开始计时
        TF1 = 0;                                                //清除TF1标志
//        TR1 = 1;                                                //定时器1开始计时
        EA=1;                                                        //开总中断
        ET1=1;                                                //开定时器1中断
}

/*************************
          按键扫描程序
*************************/
void key_scan()
{
        static bit key1_sign,key4_sign;                        //按键自锁标志变量
        static uint count1,count2,count3=0,count4=0;//消抖计数变量                       

        if(!K1)                                                        //检测按键1按下
        {
                count1++;                                        //消抖计数1自+1
                if((count1>=key_S)&&(key1_sign==0))//检测消抖计数与按键1自锁标志
                {                       
                        key1_sign=1;                        //按键1自锁标志置1
                        Choice++;                                //调整选择变量自+1
                        if(Choice>=4)                        //调整时间选择0正常走时,1调时,2调分,3调秒
                        {
                                Choice=0;                        //调整时间选择清0
                                TF0=0;                                //定时器溢出标志TF0清0
                                Cnt100us=0;                        //时间变量Cnt100us自+1
                        }
                }
        }
        else
        {
                key1_sign=0;                                //按键1自锁标志清0
                count1=0;                                        //消抖计数count1清0
        }
                               
        if(!K2)   
        {  
                count2++;  
                if(count2>=key_L)                        //长按快调
                {
                        if(Choice==1)                        //选择变量1调时
                        {
                                hour++;
                                if(hour>=24)
                                        hour=0;
                        }
                        if(Choice==2)                        //选择变量2调分
                        {
                                min++;
                                if(min>=60)
                                        min=0;
                        }
                        if(Choice==3)                        //选择变量3调秒
                        {
                                sec++;
                                if(sec>=60)
                                        sec=0;
                        }
                        if(Choice4==1)                        //选择变量1调时
                        {
                                hour4++;
                                if(hour4>=24)
                                        hour4=0;
                        }
                        if(Choice4==2)                        //选择变量2调分
                        {
                                min4++;
                                if(min4>=60)
                                        min4=0;
                        }
                        count2=key_M;
                }
        }  
        else                                                        //按键抬起
        {  
                if(count2>key_S && count2<key_L)//短按
                {
                        if(Choice==1)                        //选择变量1调时
                        {
                                hour++;
                                if(hour>=24)
                                        hour=0;
                        }
                        if(Choice==2)                        //选择变量2调分
                        {
                                min++;
                                if(min>=60)
                                        min=0;
                        }
                        if(Choice==3)                        //选择变量3调秒
                        {
                                sec++;
                                if(sec>=60)
                                        sec=0;
                        }
                        if(Choice4==1)                        //选择变量1调时
                        {
                                hour4++;
                                if(hour4>=24)
                                        hour4=0;
                        }
                        if(Choice4==2)                        //选择变量2调分
                        {
                                min4++;
                                if(min4>=60)
                                        min4=0;
                        }
                }
                count2=0;                                        //count2清0
        }   
        if(!K3)   
        {  
                count3++;  
                if(count3>=key_L)                        //长按
                {
                        if(Choice==1)                        //选择变量
                        {
                                hour--;
                                if(hour>=24)
                                        hour=23;
                        }
                        if(Choice==2)                        //选择变量
                        {
                                min--;
                                if(min>=60)
                                        min=59;
                        }
                        if(Choice==3)                        //选择变量
                        {
                                sec--;
                                if(sec>=60)
                                        sec=59;
                        }
                        if(Choice4==1)                        //选择变量
                        {
                                hour4--;
                                if(hour4>=24)
                                        hour4=23;
                        }
                        if(Choice4==2)                        //选择变量
                        {
                                min4--;
                                if(min4>=60)
                                        min4=59;
                        }
                        count3=key_M;
                }
        }  
        else                                                        //按键抬起
        {  
                if(count3>key_S && count3<key_L)//短按
                {
                        if(Choice==1)                        //选择变量
                        {
                                hour--;
                                if(hour>=24)
                                        hour=23;
                        }
                        if(Choice==2)                        //选择变量
                        {
                                min--;
                                if(min>=60)
                                        min=59;
                        }
                        if(Choice==3)                        //选择变量
                        {
                                sec--;
                                if(sec>=60)
                                        sec=59;
                        }
                        if(Choice4==1)                        //选择变量
                        {
                                hour4--;
                                if(hour4>=24)
                                        hour4=23;
                        }
                        if(Choice4==2)                        //选择变量
                        {
                                min4--;
                                if(min4>=60)
                                        min4=59;
                        }
                }
                count3=0;                                        //count3清0
        }   
        if(!K4)                                                        //检测按键1按下
        {
                count4++;                                        //消抖计数1自+1
                if((count4>=key_S)&&(key4_sign==0))//检测消抖计数与按键1自锁标志
                {                       
                        key4_sign=1;                        //按键1自锁标志置1
                        Choice4++;                                //调整选择变量自+1
                        if(Choice4>=4)                        //调整闹钟时间选择0闹钟关闭,1调时,2调分,3开启闹钟
                        {
                                Choice4=0;                        //调整时间选择清0
                        }
                }
        }
        else
        {
                key4_sign=0;                                //按键1自锁标志清0
                count4=0;                                        //消抖计数count1清0
        }
}
/************************************
        计时子程序
************************************/
void Time()       
{
        if(TF0==1)                                                //如果查询定时器溢出标志TF0为1
        {                                                                //定时器溢出周期100us
                TF0=0;                                                //定时器溢出标志TF0清0
                Cnt100us++;                                        //时间变量Cnt100us自+1
                if((Cnt100us>2500 && Cnt100us<5000)||(Cnt100us>7500 && Cnt100us<10000))//闪烁频率2Hz               
                        Twinkle=1;                                //闪烁标志
                else Twinkle=0;
                if(Cnt100us>=9982)                    //在此可以按万分之一秒调整精度
                {
                        Cnt100us=0;                                //变量Cnt100us清0
                        second=1;
                        if(Choice!=3)                        //调整选择变量为3停止走秒
                                sec++;                                //秒自+1
                        if(sec>=60)                                //如果秒>=60
                        {
                                sec=0;                                //秒清0
                                min++;                                //分自+1
                                if(min>=60)                        //分>=60
                                {
                                        min=0;                        //分清0
                                        hour++;                  //小时自+1
                                        if(hour>=24)        //小时>=24
                                                hour=0;                //小时清0
                                }
                        }
                }
        }
}
/********************************
                数码管显示程序
********************************/
void display()
{
        static uchar num=0;                                //分时显示变量
        static uchar num1=0;                                //计数延时变量
        num1++;       
        if(num1>=30)                                                //1~255可调,数码管闪烁可减小,有鬼影可加大
        {
                num1=0;
                if((Choice4>0)&&(Choice4<3))
                {
                        switch(num)
                        {
                                case 0:
                                        P0=table[hour4/10];        //时十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0x7e;                        //时十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 1:
                                        if((Twinkle==1)&&(Choice4==1))//时点闪烁
                                                P0=table[hour4%10];
                                        else
                                                P0=table[hour4%10]|0x80;//时个位段码+点
                                        dula=1;
                                        dula=0;
                                        P0=0x7d;                        //时个位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 2:
                                        P0=table[min4/10];        //分十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0x7b;                        //分十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 3:
                                        if((Twinkle==1)&&(Choice4==2))//分点闪烁
                                                P0=table[min4%10];
                                        else
                                                P0=table[min4%10]|0x80;//分个位段码+点
                                        dula=1;
                                        dula=0;
                                        P0=0x77;                        //分个位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       

                                case 4:
                                        P0=0x40;        //秒十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0xef;                        //秒十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 5:
                                        P0=0x40;//秒个位段码
                                        dula=1;
                                        dula=0;
                                        P0=0xdf;                        //秒个位位码
                                        wela=1;
                                        wela=0;
                                        num=0;
                                 break;       
                        }
                }
                else
                {
                        switch(num)
                        {
                                case 0:
                                        P0=table[hour/10];        //时十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0x7e;                        //时十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 1:
                                        if((Twinkle==1)&&(Choice==1))//时点闪烁
                                                P0=table[hour%10];
                                        else
                                                P0=table[hour%10]|0x80;//时个位段码+点
                                        dula=1;
                                        dula=0;
                                        P0=0x7d;                        //时个位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 2:
                                        P0=table[min/10];        //分十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0x7b;                        //分十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 3:
                                        if((Twinkle==1)&&(Choice==2))//分点闪烁
                                                P0=table[min%10];
                                        else
                                                P0=table[min%10]|0x80;//分个位段码+点
                                        dula=1;
                                        dula=0;
                                        P0=0x77;                        //分个位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 4:
                                        P0=table[sec/10];        //秒十位段码
                                        dula=1;
                                        dula=0;
                                        P0=0xef;                        //秒十位位码
                                        wela=1;
                                        wela=0;
                                        num++;
                                 break;       
               
                                case 5:
                                        if(Choice4==3)
                                                P0=table[sec%10]|0x80;//闹钟秒个位+点
                                        else if((Twinkle==1)&&(Choice==3))//秒点闪烁+点
                                                P0=table[sec%10]|0x80;
                                        else
                                                P0=table[sec%10];//秒个位段码
                                        dula=1;
                                        dula=0;
                                        P0=0xdf;                        //秒个位位码
                                        wela=1;
                                        wela=0;
                                        num=0;
                                 break;       
                        }
                }
        }
}
/********************************
                闹钟程序
********************************/
void Buzzer_nz()
{
        if((Choice4==3)&&(hour4==hour)&&(min4==min)&&(sec<30))
                Buzzer_sign=1;
        else Buzzer_sign=0;
        if((second==1)&&(Buzzer_sign==1))        //计时周期1s       
        {
                TR1 = 1;                                        //定时器1开,蜂鸣器响一下
                second=0;                                        //定时周期1s变量清0
        }
}
/********************************
                主程序
********************************/
void main(void)
{
        Timer_Init();                                        //初始化定时器
        while(1)
        {
                key_scan();                                        //按键扫描
                Time();                                                //计时
                display();                                        //显示
                Buzzer_nz();                                //闹钟
        }
}

/*-----------------------------
  定时器2中断服务程序  250微秒
  (无源蜂鸣器驱动程序)
------------------------------*/
void timer1() interrupt 3
{
        static uint count3=0;                        //中断计数变量
        count3++;                                                //中断计数变量count3自+1
        Buzzer=~Buzzer;                                        //蜂鸣器端口取反
        if(count3>=500)                                        //0.1秒时间到500
        {
                count3=0;                                        //计数清0                               
                Buzzer=1;                                        //蜂鸣器端口清0
                TR1 = 0;                                                //定时器1关闭
        }
}
回复

使用道具 举报

7#
zl2168 发表于 2018-9-4 07:24 | 只看该作者
本帖最后由 zl2168 于 2018-9-4 10:25 编辑

实例94  模拟电子钟(由80C51定时器产生秒时基)
Proteus仿真一下,确认有效。
实例94 模拟电子钟(由80C51定时器产生秒时基).rar (43.05 KB, 下载次数: 4)

以上摘自张志良编著《80C51单片机仿真设计实例教程——基于Keil CProteus》清华大学出版社ISBN 978-7-302-41682-1内有常用的单片机应用100案例,用于仿真实验操作,电路与程序真实可靠可信可行。书中电路和程序设计有详细说明,程序语句条条有注解。
回复

使用道具 举报

8#
duker 发表于 2018-9-4 10:08 | 只看该作者
改定时器初值
回复

使用道具 举报

9#
zhou606 发表于 2018-9-4 10:23 | 只看该作者
用RTC芯片吧
回复

使用道具 举报

10#
416524269 发表于 2018-9-4 10:56 | 只看该作者
我的经验是首先要做好定时器中断周期的精确配置,方法是用示波器来精确地配置定时器溢出值。

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

11#
angmall 发表于 2018-9-4 23:23 | 只看该作者
单片机时钟可分为2种:一是应用时钟芯片;二是用机内时钟。应用时钟芯片肯定比机内时钟精准,它取决于时钟芯片的晶振频率是否精准。应用机内时钟一般是学生实验,让学生学习定时器中断、计时计次,无实用意义。

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

12#
luke_you 发表于 2018-9-5 09:12 | 只看该作者
我也经常遇到这样的问题,改定时定时时间大一点

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

13#
1716513132 发表于 2018-9-14 21:10 | 只看该作者
支持楼主,支持
回复

使用道具 举报

14#
aini123456789 发表于 2018-9-27 18:26 来自手机 | 只看该作者
接地的线离近一点,因为在传送的过程中其实是有微弱的信号是接受不到的,除非要用三极管,放大电路

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

15#
215661599 发表于 2018-9-27 21:15 | 只看该作者
可以试试GPS自动校时哈,我以前就折腾过。用GPS接收模块,串口读取GPS数据中的授时时间。一秒读取一次,改变一次显示就行。缺点是要放在有GPS信号的地方,或者把天线放到室外

评分

参与人数 1黑币 +20 收起 理由
admin + 20 回帖助人的奖励!

查看全部评分

回复

使用道具 举报

16#
大米861 发表于 2018-9-28 08:07 | 只看该作者
选择精密的晶振。
回复

使用道具 举报

17#
@小明 发表于 2018-10-15 19:28 | 只看该作者
我的想法是把21.5秒分到24小时中去。意思是24*60分钟,21.5s/24*60    这个数值就为每分钟的补偿值,定时器定时到1分钟时就加上这个补偿值。     这种做法能相对补偿误差时间,原理是单片机时钟误差相对来说单位时间内是定值,故可以用这种方法补偿。
回复

使用道具 举报

18#
@小明 发表于 2018-10-15 19:32 | 只看该作者
我的想法是补偿。   意思是24小时差21.5s,我们可以在一天中把21.5s补偿给每一分钟。   21.5s/(24*60)   单片机定时到一分钟就把显示的时间加这个值做到补偿。    这样做的原理是因为单片机时钟误差一般相对来说是单位时间内固定值,故可以做到补偿。
回复

使用道具 举报

19#
磁通量chan1 发表于 2018-10-16 09:43 | 只看该作者
从硬件晶振那边查起吧,包括晶振的匹配电容。
回复

使用道具 举报

20#
随风飘远 发表于 2018-10-16 11:14 | 只看该作者
直接用RTC实时时钟芯片精度有保障,单片机定时器在精度要求不高的时候用。
回复

使用道具 举报

21#
ahshmj 发表于 2018-10-16 17:24 | 只看该作者
造梦Sir 发表于 2018-9-4 08:37
听你的意思是这个时钟是基于单片机定时器走的,如果真的是这样走了24小时误差这么多不足为奇,如果你要求精 ...

使用单片机做的时钟可以比时钟芯片的更容易调准。
回复

使用道具 举报

22#
ahshmj 发表于 2018-10-17 08:42 | 只看该作者
如果是单片机定时器计时的时钟,可以增加定时器的重装值。例如:假设你是用的是12M晶振,原来的重装值是(65535-50000,每秒中断20次),如下计算调整重装值:(21.5*1000*1000)/24/3600=每秒快12.44ns,(65535-50000+12)即可(其他频率的晶振照此方法计算)。(调整后理论上每天误差快38ms,如果还想再准确一些,程序中每月减掉1ms。其实单片机时钟的晶振精确度受温度等环境参数的影响,不可能精确到这)。
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

QQ|手机版|小黑屋|单片机论坛 |51黑电子论坛单片机.

Powered by 单片机教程网

快速回复 返回顶部 返回列表