如果不使用專用時鐘芯片而是使用定時器做時基,單片機時鐘的精準度取決于晶振的精準度和編程技巧?梢允谷照`差控制在1秒。給你一個示例參考,基于TX-1C實驗板。
//K1鍵調(diào)整選擇,K2鍵+,長按連+,K3鍵-,長按連-,K4鍵鬧鐘設(shè)置
//定時器初始化程序根據(jù)晶振頻率選擇
//主程序循環(huán)一次必須小于100us,否則要更改定時器周期
//用計數(shù)法代替軟件延時,提高走時精度
//數(shù)碼管采用分時動態(tài)顯示,約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; //調(diào)整選擇/退出
sbit K2 = P3^5; //++,長按連+
sbit K3 = P3^6; //--,長按連-
sbit K4 = P3^7; //調(diào)整時間
sbit dula=P2^6; //段選
sbit wela=P2^7; //位選
sbit Buzzer=P2^3; //蜂鳴器
uchar code table[]={ //0~F數(shù)組
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; //調(diào)整時間、鬧鐘選擇變量
bit Buzzer_sign; //蜂鳴器鬧鐘標志
bit Twinkle,second=1; //閃爍標志、秒標志
/**************************************
定時器0初始化程序 100微秒@12.000MHz
**************************************/
/*
void Timer0Init() //100微秒@12.000MHz
{
TMOD = 0x02; //設(shè)置自動重載模式
TL0 = 0x9C; //設(shè)置定時初值
TH0 = 0x9C; //設(shè)置定時重載值
TF0 = 0; //清除TF0標志
TR0 = 1; //定時器0開始計時
}
*/
/***************************************
定時器0/1初始化程序 100微秒@11.0592MHz
***************************************/
void Timer_Init() //100微秒@11.0592MHz
{
TMOD = 0x22; //設(shè)置自動重載模式
TL0 = 0xA4; //設(shè)置定時初值
TH0 = 0xA4; //設(shè)置定時重載值
TL1 = 0x1A; //設(shè)置定時初值250微秒@11.0592MHz
TH1 = 0x1A; //設(shè)置定時重載值
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;//消抖計數(shù)變量
if(!K1) //檢測按鍵1按下
{
count1++; //消抖計數(shù)1自+1
if((count1>=key_S)&&(key1_sign==0))//檢測消抖計數(shù)與按鍵1自鎖標志
{
key1_sign=1; //按鍵1自鎖標志置1
Choice++; //調(diào)整選擇變量自+1
if(Choice>=4) //調(diào)整時間選擇0正常走時,1調(diào)時,2調(diào)分,3調(diào)秒
{
Choice=0; //調(diào)整時間選擇清0
TF0=0; //定時器溢出標志TF0清0
Cnt100us=0; //時間變量Cnt100us自+1
}
}
}
else
{
key1_sign=0; //按鍵1自鎖標志清0
count1=0; //消抖計數(shù)count1清0
}
if(!K2)
{
count2++;
if(count2>=key_L) //長按快調(diào)
{
if(Choice==1) //選擇變量1調(diào)時
{
hour++;
if(hour>=24)
hour=0;
}
if(Choice==2) //選擇變量2調(diào)分
{
min++;
if(min>=60)
min=0;
}
if(Choice==3) //選擇變量3調(diào)秒
{
sec++;
if(sec>=60)
sec=0;
}
if(Choice4==1) //選擇變量1調(diào)時
{
hour4++;
if(hour4>=24)
hour4=0;
}
if(Choice4==2) //選擇變量2調(diào)分
{
min4++;
if(min4>=60)
min4=0;
}
count2=key_M;
}
}
else //按鍵抬起
{
if(count2>key_S && count2<key_L)//短按
{
if(Choice==1) //選擇變量1調(diào)時
{
hour++;
if(hour>=24)
hour=0;
}
if(Choice==2) //選擇變量2調(diào)分
{
min++;
if(min>=60)
min=0;
}
if(Choice==3) //選擇變量3調(diào)秒
{
sec++;
if(sec>=60)
sec=0;
}
if(Choice4==1) //選擇變量1調(diào)時
{
hour4++;
if(hour4>=24)
hour4=0;
}
if(Choice4==2) //選擇變量2調(diào)分
{
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++; //消抖計數(shù)1自+1
if((count4>=key_S)&&(key4_sign==0))//檢測消抖計數(shù)與按鍵1自鎖標志
{
key4_sign=1; //按鍵1自鎖標志置1
Choice4++; //調(diào)整選擇變量自+1
if(Choice4>=4) //調(diào)整鬧鐘時間選擇0鬧鐘關(guān)閉,1調(diào)時,2調(diào)分,3開啟鬧鐘
{
Choice4=0; //調(diào)整時間選擇清0
}
}
}
else
{
key4_sign=0; //按鍵1自鎖標志清0
count4=0; //消抖計數(shù)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) //在此可以按萬分之一秒調(diào)整精度
{
Cnt100us=0; //變量Cnt100us清0
second=1;
if(Choice!=3) //調(diào)整選擇變量為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
}
}
}
}
}
/********************************
數(shù)碼管顯示程序
********************************/
void display()
{
static uchar num=0; //分時顯示變量
static uchar num1=0; //計數(shù)延時變量
num1++;
if(num1>=30) //1~255可調(diào),數(shù)碼管閃爍可減小,有鬼影可加大
{
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中斷服務(wù)程序 250微秒
(無源蜂鳴器驅(qū)動程序)
------------------------------*/
void timer1() interrupt 3
{
static uint count3=0; //中斷計數(shù)變量
count3++; //中斷計數(shù)變量count3自+1
Buzzer=~Buzzer; //蜂鳴器端口取反
if(count3>=500) //0.1秒時間到500
{
count3=0; //計數(shù)清0
Buzzer=1; //蜂鳴器端口清0
TR1 = 0; //定時器1關(guān)閉
}
} |