【C51】源碼 3 -- 電子時鐘(獨立按鍵、8 位數(shù)碼管、定時器)
注:由于使用了中斷以及各種延時程序,此程序并不是真正準確的“電子時鐘”,想要真正準確,需要特別的外部模塊,如時鐘芯片 DS1302。
本文引用地址:http://2s4d.com/article/201611/315359.htm附上源碼:(初出茅廬,難免有寫的不好的地方,僅作備份之用,歡迎指點,噴子退散……)
/**********************************************************************************
電子時鐘
1、初始從 00-00-00 時刻開始顯示
2、按 Pause 鍵暫停,此時可以通過三個按鍵調節(jié)時間
3、再按 Pause 鍵恢復工作
缺陷:
1、按鍵過程中,有瞬間閃爍
2、再次按 Pause 開始時,會等待一會兒,電子時鐘才會工作
***********************************************************************************/
#include
#defineDATA P1// 數(shù)碼管數(shù)據(jù)傳輸口
sbitSEG = P0^0;// 段碼輸入
sbitBIT = P0^2;// 位碼輸入
sbitKEY_PAUSE = P2^0;// 暫停 / 開始 按鍵
sbitKEY_H_UP = P2^2;// 小時調節(jié) 按鍵
sbitKEY_M_UP = P2^4;// 分鐘調節(jié) 按鍵
sbitKEY_S_UP = P2^6;// 秒調節(jié) 按鍵
unsigned char codeSEG_CODE[] = {0x3f, 0x06, 0x5b, 0x4f, 0x66, 0x6d, 0x7d, 0x07, 0x7f, 0x6f, 0x40};
unsigned char codeBIT_CODE[] = {0xfe, 0xfd, 0xfb, 0xf7, 0xef, 0xdf, 0xbf, 0x7f};
// 上面是段碼表,0-9,還有一個 “-” 符號
// 下面是位碼表,指示哪一位顯示字符
unsigned chartimer0_count;// 定時器 50ms 一個中斷,計 20 次為 1s
unsigned charH1, H0, M1, M0, S1, S0;// 電子時鐘顯示數(shù)字 HH-MM-SS
voidDelay(unsigned intt);// 普通延遲函數(shù)
voidDelay_ms(unsigned intt);// ms 延遲函數(shù)
voidInit_Key(void);// 初始化按鍵
voidInit_Clock(void);// 初始化電子時鐘顯示
voidInit_Timer0(void);// 初始化定時器
voidDisplay(unsigned charx,unsigned charnum);// 在 x 位置,顯示數(shù)字 num
voidDisplay_Clock(void);// 顯示電子時鐘
voidmain (void)
{
Init_Key();
Init_Clock();
Init_Timer0();
while(1) {
Display_Clock();// 在大循環(huán)中顯示時鐘,形成動態(tài)掃描
if(timer0_count == 20) {// 定時器中斷了 20 次,即 1s,開始對顯示進行處理
timer0_count = 0;
S0 += 1;
if(S0 == 10) {// 以下保證了秒:0-59,分鐘:0-59,小時:0-23
S0 = 0;
S1 += 1;
}
if(S1 == 6) {
S1 = 0;
M0 +=1;
}
if(M0 == 10) {
M0 = 0;
M1 += 1;
}
if(M1 == 6) {
M1 = 0;
H0 += 1;
}
if(H0 == 10 && (H1 == 0 || H1 == 1)) {
H0 = 0;
H1 += 1;
}
else if(H0 == 4 && H1 == 2) {
H0 = 0;
H1 = 0;
}
}
if(!KEY_PAUSE) { // 檢測 暫停鍵 是否按下
Delay_ms(10);// 延遲 10ms,消除按鍵抖動,后邊同理
if(!KEY_PAUSE) {// 在延遲之后,還能檢測到按鍵,說明真的是按鍵
while(!KEY_PAUSE) Display_Clock();// 如果按住按鍵不放,就保持時鐘顯示狀態(tài)
while(KEY_PAUSE) {// 按鍵終于放開了,處理按鍵的行為
Display_Clock();// 在此期間,要保持時鐘一直動態(tài)顯示
if(!KEY_H_UP) { // 在暫停期間,檢測到了 小時調節(jié) 操作,處理顯示
Delay_ms(10);
if(!KEY_H_UP) {
while(!KEY_H_UP) Display_Clock();
H0 += 1;
if(H0 == 10 && (H1 == 0 || H1 == 1)) {
H0 = 0;
H1 += 1;
}
else if(H0 == 4 && H1 == 2) {
H0 = 0;
H1 = 0;
}
}
}
if(!KEY_M_UP) {// 在暫停期間,檢測到了 分鐘調節(jié) 操作,處理顯示
Delay_ms(10);
if(!KEY_M_UP) {
while(!KEY_M_UP) Display_Clock();
M0 += 1;
if(M0 == 10) {
M0 = 0;
M1 += 1;
}
if(M1 == 6) M1 = 0;
}
}
if(!KEY_S_UP) {// 在暫停期間,檢測到了 秒調節(jié) 操作,處理顯示
Delay_ms(10);
if(!KEY_S_UP) {
while(!KEY_S_UP) Display_Clock();
S0 += 1;
if(S0 == 10) {
S0 = 0;
S1 += 1;
}
if(S1 == 6) S1 = 0;
}
}
}
if(!KEY_PAUSE) {// 再次按下 暫停 按鍵,恢復工作
Delay_ms(10);
if(!KEY_PAUSE) {
while(!KEY_PAUSE) Display_Clock();
}
}
}
}
}
}
voidDelay(unsigned intt)
{
while(t--);
}
voidDelay_ms(unsigned intt)// 根據(jù)測試,可以相當近似的表示 1ms
{
while(t--) {
Delay(245);
Delay(245);
}
}
voidInit_Key(void)
{
KEY_PAUSE = 1;// 賦值高電平為 按鍵斷開 狀態(tài)
KEY_H_UP = 1;
KEY_M_UP = 1;
KEY_S_UP = 1;
}
voidInit_Clock(void)// 初始時鐘顯示為 00-00-00
{
H1 = 0;
H0 = 0;
M1 = 0;
M0 = 0;
S1 = 0;
S0 = 0;
}
voidInit_Timer0(void)
{
TMOD = 0x01;// 使用定時器 0,模式 1
EA = 1;// 開總中斷
ET0 = 1;// 開定時器 0 中斷
TH0 = (65536-50000)/256;// 定時器 0 賦值,定時 50ms
TL0 = (65536-50000)%256;
TR0 = 1;// 定時器 0 開始工作
timer0_count = 0;// 計中斷個數(shù),目標 20
}
/*************************************************************************************************
當定時器 0 定時完畢,進入此中斷過程,后邊的 1 代表定時器 0 的中斷
*************************************************************************************************/
voidTimer0_ISR(void)interrupt1
{
TH0 = (65536-50000)/256;
TL0 = (65536-50000)%256;
timer0_count++;
}
voidDisplay(unsigned charx,unsigned charnum)
{
SEG = 1;// 關鎖存器
DATA = SEG_CODE[num];// 送段碼
SEG = 0;// 開鎖存器
BIT = 1;// 關鎖存器
DATA = BIT_CODE[x-1];// 送位碼
BIT = 0;// 開鎖存器
Delay(200);// 延時,形成動態(tài)顯示,200 正好,過大閃爍,過小重影
}
voidDisplay_Clock(void)
{
Display(1, H1);// 在 1、2 位置顯示 小時
Display(2, H0);
Display(3, 10);
Display(4, M1);// 在 4、5 位置顯示 分鐘
Display(5, M0);
Display(6, 10);
Display(7, S1);// 在 7、8 位置顯示 秒,其他位置顯示 “-”
Display(8, S0);
}
評論