51單片機(jī)T0做精確時鐘
有兩個方法可以解決這個問題。
本文引用地址:http://m.butianyuan.cn/article/201611/318363.htm第一個方法:你可以計算出中斷處理時重新給定時器賦初值所用的機(jī)器周期數(shù),在你計算出的初值里除去這幾個機(jī)器周期,作為補(bǔ)償。這個方法,只是在每次定時器中斷后,都可以按時得到執(zhí)行時很精確,但事實上中斷什么時候執(zhí)行誰都說不準(zhǔn),所以這個方法,只能做到盡量精確。
第二個方法:計算出一個湊巧的初值,使TL0正好等于0x00,這樣每次中斷溢出后,TL0都從0x00開始計數(shù),即使中斷沒有得到執(zhí)行,TL0也會繼續(xù)計數(shù)。利用這一點,在中斷處理函數(shù)中,只需要對TH0重新賦值,不需要管TL0。以下是示例程序:
//定時器T0時鐘參數(shù)
unsigned char T0_S = 0;//秒
unsigned char T0_M = 0;//分
unsigned char T0_H = 0;//時
unsigned char T0_Cycle = 0;//循環(huán)次數(shù)
//本程序所用晶振22.1184MHz,每次定時25ms,循環(huán)40次正好1s。
void Timer0_Init(void)//T0初始化函數(shù)
{
TMOD = 0x01;//設(shè)置T0工作方式1
TH0= 0x4c;//(65536-46080)/256,設(shè)置初值46080,晶振22.1184MHz,
//每個機(jī)器周期0.5425微秒,定時25ms
TL0= 0x00;//(65536-46080)%256,TL0恰好=0x00
IE|= 0x82;//開中斷
TR0= 1 ;//T0開始定時
}
//
void Timer0(void) interrupt 1
{
TH0= 0x4c;//重新給TH0賦值
//TL0= 0x00;//不對TL0賦值,讓其繼續(xù)計數(shù)
TF0= 0 ;//定時溢出清0
T0_Cycle ++ ;
if(T0_Cycle == 40)//循環(huán)40次,每次25ms,定時1s
{
T0_Cycle = 0 ;
T0_S++ ;
if(T0_S == 60)
{
T0_S = 0;
T0_M++;
if(T0_M == 60)
{
T0_M = 0;
T0_H++;
if(T0_H == 24)
{
T0_H = 0;
}
}
}
}
}
以上程序即使T0中斷賦初值會耽誤幾個機(jī)器周期,中斷沒有得到及時執(zhí)行也沒有關(guān)系,因為TL0的計數(shù)不受影響。但有一種情況必須要注意,雖然這種情況發(fā)生的可能性不大。如果T0中斷長時間沒有得到響應(yīng),TL0再一次溢出了,這時這種方法的誤差就大了。
經(jīng)過測試,第二種方法還是比較精確的,時鐘跑了一天快了10s,這個誤差應(yīng)該是晶振本身的問題,如果是定時器的問題,應(yīng)該是慢了,不會快了。
評論