第47節(jié):操作AT24C02利用定時器延時改善數(shù)碼管的閃爍
上一節(jié)在按鍵更改參數(shù)時,會出現(xiàn)短暫明顯的數(shù)碼管閃爍現(xiàn)象。這節(jié)通過教大家使用新型延時函數(shù)可以有效的改善閃爍現(xiàn)象。要教會大家三個知識點:
第一個:如何編寫一氣呵成的定時器延時函數(shù)。
第二個:如何編寫檢查EEPROM芯片是否存在短路,虛焊或者芯片壞了的監(jiān)控程序。
第三個:經(jīng)過網(wǎng)友“cjseng”的提醒,我建議大家以后在用EEPROM芯片時,如果單片機IO口足夠多,WP引腳應該專門接一個IO口,并且加一個上拉電阻,需要更改EEPROM存儲數(shù)據(jù)時置低,其他任何一個時刻都置高,這樣可以更加有效地保護EEPROM內(nèi)部數(shù)據(jù)不會被意外更改。
具體內(nèi)容,請看源代碼講解。
(1)硬件平臺:
基于朱兆祺51單片機學習板。舊版的朱兆祺51學習板在硬件上有一個bug,AT24C02的第8個引腳VCC懸空了!!!,讀者記得把它飛線連接到5V電源處。新版的朱兆祺51學習板已經(jīng)改過來了。
(2)實現(xiàn)功能:
4個被更改后的參數(shù)斷電后不丟失,數(shù)據(jù)可以保存,斷電再上電后還是上一次最新被修改的數(shù)據(jù)。如果AT24C02短路,虛焊,或者壞了,系統(tǒng)可以檢查出來,并且蜂鳴器會間歇性鳴叫報警。按更改參數(shù)按鍵時,數(shù)碼管比上一節(jié)大大降低了閃爍現(xiàn)象。
顯示和獨立按鍵部分根據(jù)第29節(jié)的程序來改編,用朱兆祺51單片機學習板中的S1,S5,S9作為獨立按鍵。
一共有4個窗口。每個窗口顯示一個參數(shù)。
第8,7,6,5位數(shù)碼管顯示當前窗口,P-1代表第1個窗口,P-2代表第2個窗口,P-3代表第3個窗口,P-4代表第1個窗口。
第4,3,2,1位數(shù)碼管顯示當前窗口被設置的參數(shù)。范圍是從0到9999。S1是加按鍵,按下此按鍵會依次增加當前窗口的參數(shù)。S5是減按鍵,按下此按鍵會依次減少當前窗口的參數(shù)。S9是切換窗口按鍵,按下此按鍵會依次循環(huán)切換不同的窗口。
(3)源代碼講解如下:
- #include "REG52.H"
- #define const_voice_short40 //蜂鳴器短叫的持續(xù)時間
- #define const_key_time120 //按鍵去抖動延時的時間
- #define const_key_time220 //按鍵去抖動延時的時間
- #define const_key_time320 //按鍵去抖動延時的時間
- #define const_eeprom_1s 400 //大概1秒的時間
- void initial_myself(void);
- void initial_peripheral(void);
- void delay_short(unsigned int uiDelayShort);
- void delay_long(unsigned int uiDelaylong);
- void delay_timer(unsigned int uiDelayTimerTemp); //一氣呵成的定時器延時方式
- //驅(qū)動數(shù)碼管的74HC595
- void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);
- void display_drive(void); //顯示數(shù)碼管字模的驅(qū)動函數(shù)
- void display_service(void); //顯示的窗口菜單服務程序
- //驅(qū)動LED的74HC595
- void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01);
- void start24(void);//開始位
- void ack24(void);//確認位
- void stop24(void);//停止位
- unsigned char read24(void);//讀取一個字節(jié)的時序
- void write24(unsigned char dd); //發(fā)送一個字節(jié)的時序
- unsigned char read_eeprom(unsigned int address); //從一個地址讀取出一個字節(jié)數(shù)據(jù)
- void write_eeprom(unsigned int address,unsigned char dd); //往一個地址存入一個字節(jié)數(shù)據(jù)
- unsigned int read_eeprom_int(unsigned int address); //從一個地址讀取出一個int類型的數(shù)據(jù)
- void write_eeprom_int(unsigned int address,unsigned int uiWriteData); //往一個地址存入一個int類型的數(shù)據(jù)
- void T0_time(void);//定時中斷函數(shù)
- void key_service(void); //按鍵服務的應用程序
- void key_scan(void);//按鍵掃描函數(shù) 放在定時中斷里
- void eeprom_alarm_service(void); //EEPROM出錯報警
- sbit key_sr1=P0^0; //對應朱兆祺學習板的S1鍵
- sbit key_sr2=P0^1; //對應朱兆祺學習板的S5鍵
- sbit key_sr3=P0^2; //對應朱兆祺學習板的S9鍵
- sbit key_gnd_dr=P0^4; //模擬獨立按鍵的地GND,因此必須一直輸出低電平
- sbit beep_dr=P2^7; //蜂鳴器的驅(qū)動IO口
- sbit eeprom_scl_dr=P3^7; //時鐘線
- sbit eeprom_sda_dr_sr=P3^6; //數(shù)據(jù)的輸出線和輸入線
- sbit dig_hc595_sh_dr=P2^0; //數(shù)碼管的74HC595程序
- sbit dig_hc595_st_dr=P2^1;
- sbit dig_hc595_ds_dr=P2^2;
- sbit hc595_sh_dr=P2^3; //LED燈的74HC595程序
- sbit hc595_st_dr=P2^4;
- sbit hc595_ds_dr=P2^5;
- unsigned char ucKeySec=0; //被觸發(fā)的按鍵編號
- unsigned intuiKeyTimeCnt1=0; //按鍵去抖動延時計數(shù)器
- unsigned char ucKeyLock1=0; //按鍵觸發(fā)后自鎖的變量標志
- unsigned intuiKeyTimeCnt2=0; //按鍵去抖動延時計數(shù)器
- unsigned char ucKeyLock2=0; //按鍵觸發(fā)后自鎖的變量標志
- unsigned intuiKeyTimeCnt3=0; //按鍵去抖動延時計數(shù)器
- unsigned char ucKeyLock3=0; //按鍵觸發(fā)后自鎖的變量標志
- unsigned intuiVoiceCnt=0;//蜂鳴器鳴叫的持續(xù)時間計數(shù)器
- unsigned charucVoiceLock=0;//蜂鳴器鳴叫的原子鎖
- unsigned char ucDigShow8;//第8位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow7;//第7位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow6;//第6位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow5;//第5位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow4;//第4位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow3;//第3位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow2;//第2位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigShow1;//第1位數(shù)碼管要顯示的內(nèi)容
- unsigned char ucDigDot8;//數(shù)碼管8的小數(shù)點是否顯示的標志
- unsigned char ucDigDot7;//數(shù)碼管7的小數(shù)點是否顯示的標志
- unsigned char ucDigDot6;//數(shù)碼管6的小數(shù)點是否顯示的標志
- unsigned char ucDigDot5;//數(shù)碼管5的小數(shù)點是否顯示的標志
- unsigned char ucDigDot4;//數(shù)碼管4的小數(shù)點是否顯示的標志
- unsigned char ucDigDot3;//數(shù)碼管3的小數(shù)點是否顯示的標志
- unsigned char ucDigDot2;//數(shù)碼管2的小數(shù)點是否顯示的標志
- unsigned char ucDigDot1;//數(shù)碼管1的小數(shù)點是否顯示的標志
- unsigned char ucDigShowTemp=0; //臨時中間變量
- unsigned char ucDisplayDriveStep=1;//動態(tài)掃描數(shù)碼管的步驟變量
- unsigned char ucWd1Update=1; //窗口1更新顯示標志
- unsigned char ucWd2Update=0; //窗口2更新顯示標志
- unsigned char ucWd3Update=0; //窗口3更新顯示標志
- unsigned char ucWd4Update=0; //窗口4更新顯示標志
- unsigned char ucWd=1;//本程序的核心變量,窗口顯示變量。類似于一級菜單的變量。代表顯示不同的窗口。
- unsigned intuiSetData1=0;//本程序中需要被設置的參數(shù)1
- unsigned intuiSetData2=0;//本程序中需要被設置的參數(shù)2
- unsigned intuiSetData3=0;//本程序中需要被設置的參數(shù)3
- unsigned intuiSetData4=0;//本程序中需要被設置的參數(shù)4
- unsigned char ucTemp1=0;//中間過渡變量
- unsigned char ucTemp2=0;//中間過渡變量
- unsigned char ucTemp3=0;//中間過渡變量
- unsigned char ucTemp4=0;//中間過渡變量
- unsigned char ucDelayTimerLock=0; //原子鎖
- unsigned intuiDelayTimer=0;
- unsigned char ucCheckEeprom=0;//檢查EEPROM芯片是否正常
- unsigned char ucEepromError=0; //EEPROM芯片是否正常的標志
- unsigned char ucEepromLock=0;//原子鎖
- unsigned intuiEepromCnt=0; //間歇性蜂鳴器報警的計時器
- //根據(jù)原理圖得出的共陰數(shù)碼管字模表
- code unsigned char dig_table[]=
- {
- 0x3f,//0 序號0
- 0x06,//1 序號1
- 0x5b,//2 序號2
- 0x4f,//3 序號3
- 0x66,//4 序號4
- 0x6d,//5 序號5
- 0x7d,//6 序號6
- 0x07,//7 序號7
- 0x7f,//8 序號8
- 0x6f,//9 序號9
- 0x00,//無 序號10
- 0x40,//- 序號11
- 0x73,//P 序號12
- };
- void main()
- {
- initial_myself();
- delay_long(100);
- initial_peripheral();
- while(1)
- {
- key_service(); //按鍵服務的應用程序
- display_service(); //顯示的窗口菜單服務程序
- eeprom_alarm_service(); //EEPROM出錯報警
- }
- }
- void eeprom_alarm_service(void) //EEPROM出錯報警
- {
- if(ucEepromError==1) //EEPROM出錯
- {
- if(uiEepromCnt
- {
- ucEepromLock=1;//原子鎖加鎖
- uiEepromCnt=0; //計時器清零
- ucEepromLock=0;//原子鎖解鎖
- ucVoiceLock=1;//原子鎖加鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- uiVoiceCnt=const_voice_short; //蜂鳴器聲音觸發(fā),滴一聲就停。
- ucVoiceLock=0;//原子鎖解鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- }
- }
- }
- //AT24C02驅(qū)動程序
- void start24(void)//開始位
- {
- eeprom_sda_dr_sr=1;
- eeprom_scl_dr=1;
- delay_short(15);
- eeprom_sda_dr_sr=0;
- delay_short(15);
- eeprom_scl_dr=0;
- }
- void ack24(void)//確認位時序
- {
- eeprom_sda_dr_sr=1; //51單片機在讀取數(shù)據(jù)之前要先置一,表示數(shù)據(jù)輸入
- eeprom_scl_dr=1;
- delay_short(15);
- eeprom_scl_dr=0;
- delay_short(15);
- //在本驅(qū)動程序中,我沒有對ACK信號進行出錯判斷,因為我這么多年一直都是這樣用也沒出現(xiàn)過什么問題。
- //有興趣的朋友可以自己增加出錯判斷,不一定非要按我的方式去做。
- }
- void stop24(void)//停止位
- {
- eeprom_sda_dr_sr=0;
- eeprom_scl_dr=1;
- delay_short(15);
- eeprom_sda_dr_sr=1;
- }
- unsigned char read24(void)//讀取一個字節(jié)的時序
- {
- unsigned char outdata,tempdata;
- outdata=0;
- eeprom_sda_dr_sr=1; //51單片機的IO口在讀取數(shù)據(jù)之前要先置一,表示數(shù)據(jù)輸入
- delay_short(2);
- for(tempdata=0;tempdata<8;tempdata++)
- {
- eeprom_scl_dr=0;
- delay_short(2);
- eeprom_scl_dr=1;
- delay_short(2);
- outdata<<=1;
- if(eeprom_sda_dr_sr==1)outdata++;
- eeprom_sda_dr_sr=1; //51單片機的IO口在讀取數(shù)據(jù)之前要先置一,表示數(shù)據(jù)輸入
- delay_short(2);
- }
- return(outdata);
- }
- void write24(unsigned char dd) //發(fā)送一個字節(jié)的時序
- {
- unsigned char tempdata;
- for(tempdata=0;tempdata<8;tempdata++)
- {
- if(dd>=0x80)eeprom_sda_dr_sr=1;
- else eeprom_sda_dr_sr=0;
- dd<<=1;
- delay_short(2);
- eeprom_scl_dr=1;
- delay_short(4);
- eeprom_scl_dr=0;
- }
- }
- unsigned char read_eeprom(unsigned int address) //從一個地址讀取出一個字節(jié)數(shù)據(jù)
- {
- unsigned char dd,cAddress;
- cAddress=address; //把低字節(jié)地址傳遞給一個字節(jié)變量。
- EA=0; //禁止中斷
- start24(); //IIC通訊開始
- write24(0xA0); //此字節(jié)包含讀寫指令和芯片地址兩方面的內(nèi)容。
- //指令為寫指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- write24(cAddress); //發(fā)送讀取的存儲地址(范圍是0至255)
- ack24(); //發(fā)送應答信號
- start24(); //開始
- write24(0xA1); //此字節(jié)包含讀寫指令和芯片地址兩方面的內(nèi)容。
- //指令為讀指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- dd=read24(); //讀取一個字節(jié)
- ack24(); //發(fā)送應答信號
- stop24();//停止
- EA=1; //允許中斷
- delay_timer(2); //一氣呵成的定時器延時方式,在延時的時候還可以動態(tài)掃描數(shù)碼管
- return(dd);
- }
- void write_eeprom(unsigned int address,unsigned char dd) //往一個地址存入一個字節(jié)數(shù)據(jù)
- {
- unsigned char cAddress;
- cAddress=address; //把低字節(jié)地址傳遞給一個字節(jié)變量。
- EA=0; //禁止中斷
- start24(); //IIC通訊開始
- write24(0xA0); //此字節(jié)包含讀寫指令和芯片地址兩方面的內(nèi)容。
- //指令為寫指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- write24(cAddress); //發(fā)送寫入的存儲地址(范圍是0至255)
- ack24(); //發(fā)送應答信號
- write24(dd);//寫入存儲的數(shù)據(jù)
- ack24(); //發(fā)送應答信號
- stop24();//停止
- EA=1; //允許中斷
- delay_timer(4); //一氣呵成的定時器延時方式,在延時的時候還可以動態(tài)掃描數(shù)碼管
- }
- unsigned int read_eeprom_int(unsigned int address) //從一個地址讀取出一個int類型的數(shù)據(jù)
- {
- unsigned char ucReadDataH;
- unsigned char ucReadDataL;
- unsigned intuiReadDate;
- ucReadDataH=read_eeprom(address); //讀取高字節(jié)
- ucReadDataL=read_eeprom(address+1);//讀取低字節(jié)
- uiReadDate=ucReadDataH;//把兩個字節(jié)合并成一個int類型數(shù)據(jù)
- uiReadDate=uiReadDate<<8;
- uiReadDate=uiReadDate+ucReadDataL;
- return uiReadDate;
- }
- void write_eeprom_int(unsigned int address,unsigned int uiWriteData) //往一個地址存入一個int類型的數(shù)據(jù)
- {
- unsigned char ucWriteDataH;
- unsigned char ucWriteDataL;
- ucWriteDataH=uiWriteData>>8;
- ucWriteDataL=uiWriteData;
- write_eeprom(address,ucWriteDataH); //存入高字節(jié)
- write_eeprom(address+1,ucWriteDataL); //存入低字節(jié)
- }
- void display_service(void) //顯示的窗口菜單服務程序
- {
- switch(ucWd)//本程序的核心變量,窗口顯示變量。類似于一級菜單的變量。代表顯示不同的窗口。
- {
- case 1: //顯示P--1窗口的數(shù)據(jù)
- if(ucWd1Update==1)//窗口1要全部更新顯示
- {
- ucWd1Update=0;//及時清零標志,避免一直進來掃描
- ucDigShow8=12;//第8位數(shù)碼管顯示P
- ucDigShow7=11;//第7位數(shù)碼管顯示-
- ucDigShow6=1; //第6位數(shù)碼管顯示1
- ucDigShow5=10;//第5位數(shù)碼管顯示無
- //先分解數(shù)據(jù)
- ucTemp4=uiSetData1/1000;
- ucTemp3=uiSetData1%1000/100;
- ucTemp2=uiSetData1%100/10;
- ucTemp1=uiSetData1%10;
- //再過渡需要顯示的數(shù)據(jù)到緩沖變量里,讓過渡的時間越短越好
- if(uiSetData1<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData1<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData1<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內(nèi)容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內(nèi)容
- }
- break;
- case 2://顯示P--2窗口的數(shù)據(jù)
- if(ucWd2Update==1)//窗口2要全部更新顯示
- {
- ucWd2Update=0;//及時清零標志,避免一直進來掃描
- ucDigShow8=12;//第8位數(shù)碼管顯示P
- ucDigShow7=11;//第7位數(shù)碼管顯示-
- ucDigShow6=2;//第6位數(shù)碼管顯示2
- ucDigShow5=10; //第5位數(shù)碼管顯示無
- ucTemp4=uiSetData2/1000; //分解數(shù)據(jù)
- ucTemp3=uiSetData2%1000/100;
- ucTemp2=uiSetData2%100/10;
- ucTemp1=uiSetData2%10;
- if(uiSetData2<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData2<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData2<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內(nèi)容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內(nèi)容
- }
- break;
- case 3://顯示P--3窗口的數(shù)據(jù)
- if(ucWd3Update==1)//窗口3要全部更新顯示
- {
- ucWd3Update=0;//及時清零標志,避免一直進來掃描
- ucDigShow8=12;//第8位數(shù)碼管顯示P
- ucDigShow7=11;//第7位數(shù)碼管顯示-
- ucDigShow6=3;//第6位數(shù)碼管顯示3
- ucDigShow5=10; //第5位數(shù)碼管顯示無
- ucTemp4=uiSetData3/1000; //分解數(shù)據(jù)
- ucTemp3=uiSetData3%1000/100;
- ucTemp2=uiSetData3%100/10;
- ucTemp1=uiSetData3%10;
- if(uiSetData3<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData3<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData3<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內(nèi)容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內(nèi)容
- }
- break;
- case 4://顯示P--4窗口的數(shù)據(jù)
- if(ucWd4Update==1)//窗口4要全部更新顯示
- {
- ucWd4Update=0;//及時清零標志,避免一直進來掃描
- ucDigShow8=12;//第8位數(shù)碼管顯示P
- ucDigShow7=11;//第7位數(shù)碼管顯示-
- ucDigShow6=4;//第6位數(shù)碼管顯示4
- ucDigShow5=10; //第5位數(shù)碼管顯示無
- ucTemp4=uiSetData4/1000; //分解數(shù)據(jù)
- ucTemp3=uiSetData4%1000/100;
- ucTemp2=uiSetData4%100/10;
- ucTemp1=uiSetData4%10;
- if(uiSetData4<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData4<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內(nèi)容
- }
- if(uiSetData4<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內(nèi)容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內(nèi)容
- }
- break;
- }
- }
- void key_scan(void)//按鍵掃描函數(shù) 放在定時中斷里
- {
- if(key_sr1==1)//IO是高電平,說明按鍵沒有被按下,這時要及時清零一些標志位
- {
- ucKeyLock1=0; //按鍵自鎖標志清零
- uiKeyTimeCnt1=0;//按鍵去抖動延時計數(shù)器清零,此行非常巧妙,是我實戰(zhàn)中摸索出來的。
- }
- else if(ucKeyLock1==0)//有按鍵按下,且是第一次被按下
- {
- uiKeyTimeCnt1++; //累加定時中斷次數(shù)
- if(uiKeyTimeCnt1>const_key_time1)
- {
- uiKeyTimeCnt1=0;
- ucKeyLock1=1;//自鎖按鍵置位,避免一直觸發(fā)
- ucKeySec=1; //觸發(fā)1號鍵
- }
- }
- if(key_sr2==1)//IO是高電平,說明按鍵沒有被按下,這時要及時清零一些標志位
- {
- ucKeyLock2=0; //按鍵自鎖標志清零
- uiKeyTimeCnt2=0;//按鍵去抖動延時計數(shù)器清零,此行非常巧妙,是我實戰(zhàn)中摸索出來的。
- }
- else if(ucKeyLock2==0)//有按鍵按下,且是第一次被按下
- {
- uiKeyTimeCnt2++; //累加定時中斷次數(shù)
- if(uiKeyTimeCnt2>const_key_time2)
- {
- uiKeyTimeCnt2=0;
- ucKeyLock2=1;//自鎖按鍵置位,避免一直觸發(fā)
- ucKeySec=2; //觸發(fā)2號鍵
- }
- }
- if(key_sr3==1)//IO是高電平,說明按鍵沒有被按下,這時要及時清零一些標志位
- {
- ucKeyLock3=0; //按鍵自鎖標志清零
- uiKeyTimeCnt3=0;//按鍵去抖動延時計數(shù)器清零,此行非常巧妙,是我實戰(zhàn)中摸索出來的。
- }
- else if(ucKeyLock3==0)//有按鍵按下,且是第一次被按下
- {
- uiKeyTimeCnt3++; //累加定時中斷次數(shù)
- if(uiKeyTimeCnt3>const_key_time3)
- {
- uiKeyTimeCnt3=0;
- ucKeyLock3=1;//自鎖按鍵置位,避免一直觸發(fā)
- ucKeySec=3; //觸發(fā)3號鍵
- }
- }
- }
- void key_service(void) //按鍵服務的應用程序
- {
- switch(ucKeySec) //按鍵服務狀態(tài)切換
- {
- case 1:// 加按鍵 對應朱兆祺學習板的S1鍵
- switch(ucWd)//在不同的窗口下,設置不同的參數(shù)
- {
- case 1:
- uiSetData1++;
- if(uiSetData1>9999) //最大值是9999
- {
- uiSetData1=9999;
- }
- write_eeprom_int(0,uiSetData1); //存入EEPROM 由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd1Update=1;//窗口1更新顯示
- break;
- case 2:
- uiSetData2++;
- if(uiSetData2>9999) //最大值是9999
- {
- uiSetData2=9999;
- }
- write_eeprom_int(2,uiSetData2); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd2Update=1;//窗口2更新顯示
- break;
- case 3:
- uiSetData3++;
- if(uiSetData3>9999) //最大值是9999
- {
- uiSetData3=9999;
- }
- write_eeprom_int(4,uiSetData3); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd3Update=1;//窗口3更新顯示
- break;
- case 4:
- uiSetData4++;
- if(uiSetData4>9999) //最大值是9999
- {
- uiSetData4=9999;
- }
- write_eeprom_int(6,uiSetData4); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd4Update=1;//窗口4更新顯示
- break;
- }
- ucVoiceLock=1;//原子鎖加鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
- ucVoiceLock=0;//原子鎖解鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- ucKeySec=0;//響應按鍵服務處理程序后,按鍵編號清零,避免一致觸發(fā)
- break;
- case 2:// 減按鍵 對應朱兆祺學習板的S5鍵
- switch(ucWd)//在不同的窗口下,設置不同的參數(shù)
- {
- case 1:
- uiSetData1--;
- if(uiSetData1>9999)
- {
- uiSetData1=0;//最小值是0
- }
- write_eeprom_int(0,uiSetData1); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd1Update=1;//窗口1更新顯示
- break;
- case 2:
- uiSetData2--;
- if(uiSetData2>9999)
- {
- uiSetData2=0;//最小值是0
- }
- write_eeprom_int(2,uiSetData2); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd2Update=1;//窗口2更新顯示
- break;
- case 3:
- uiSetData3--;
- if(uiSetData3>9999)
- {
- uiSetData3=0;//最小值是0
- }
- write_eeprom_int(4,uiSetData3); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd3Update=1;//窗口3更新顯示
- break;
- case 4:
- uiSetData4--;
- if(uiSetData4>9999)
- {
- uiSetData4=0;//最小值是0
- }
- write_eeprom_int(6,uiSetData4); //存入EEPROM,由于內(nèi)部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd4Update=1;//窗口4更新顯示
- break;
- }
- ucVoiceLock=1;//原子鎖加鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
- ucVoiceLock=0;//原子鎖解鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- ucKeySec=0;//響應按鍵服務處理程序后,按鍵編號清零,避免一致觸發(fā)
- break;
- case 3:// 切換窗口按鍵 對應朱兆祺學習板的S9鍵
- ucWd++;//切換窗口
- if(ucWd>4)
- {
- ucWd=1;
- }
- switch(ucWd)//在不同的窗口下,在不同的窗口下,更新顯示不同的窗口
- {
- case 1:
- ucWd1Update=1;//窗口1更新顯示
- break;
- case 2:
- ucWd2Update=1;//窗口2更新顯示
- break;
- case 3:
- ucWd3Update=1;//窗口3更新顯示
- break;
- case 4:
- ucWd4Update=1;//窗口4更新顯示
- break;
- }
- ucVoiceLock=1;//原子鎖加鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- uiVoiceCnt=const_voice_short; //按鍵聲音觸發(fā),滴一聲就停。
- ucVoiceLock=0;//原子鎖解鎖,保護主函數(shù)與中斷函數(shù)的共享變量uiVoiceCnt
- ucKeySec=0;//響應按鍵服務處理程序后,按鍵編號清零,避免一致觸發(fā)
- break;
- }
- }
- void display_drive(void)
- {
- //以下程序,如果加一些數(shù)組和移位的元素,還可以壓縮容量。但是鴻哥追求的不是容量,而是清晰的講解思路
- switch(ucDisplayDriveStep)
- {
- case 1://顯示第1位
- ucDigShowTemp=dig_table[ucDigShow1];
- if(ucDigDot1==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xfe);
- break;
- case 2://顯示第2位
- ucDigShowTemp=dig_table[ucDigShow2];
- if(ucDigDot2==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xfd);
- break;
- case 3://顯示第3位
- ucDigShowTemp=dig_table[ucDigShow3];
- if(ucDigDot3==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xfb);
- break;
- case 4://顯示第4位
- ucDigShowTemp=dig_table[ucDigShow4];
- if(ucDigDot4==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xf7);
- break;
- case 5://顯示第5位
- ucDigShowTemp=dig_table[ucDigShow5];
- if(ucDigDot5==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xef);
- break;
- case 6://顯示第6位
- ucDigShowTemp=dig_table[ucDigShow6];
- if(ucDigDot6==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xdf);
- break;
- case 7://顯示第7位
- ucDigShowTemp=dig_table[ucDigShow7];
- if(ucDigDot7==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0xbf);
- break;
- case 8://顯示第8位
- ucDigShowTemp=dig_table[ucDigShow8];
- if(ucDigDot8==1)
- {
- ucDigShowTemp=ucDigShowTemp|0x80;//顯示小數(shù)點
- }
- dig_hc595_drive(ucDigShowTemp,0x7f);
- break;
- }
- ucDisplayDriveStep++;
- if(ucDisplayDriveStep>8)//掃描完8個數(shù)碼管后,重新從第一個開始掃描
- {
- ucDisplayDriveStep=1;
- }
- }
- //數(shù)碼管的74HC595驅(qū)動函數(shù)
- void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01)
- {
- unsigned char i;
- unsigned char ucTempData;
- dig_hc595_sh_dr=0;
- dig_hc595_st_dr=0;
- ucTempData=ucDigStatusTemp16_09;//先送高8位
- for(i=0;i<8;i++)
- {
- if(ucTempData>=0x80)dig_hc595_ds_dr=1;
- else dig_hc595_ds_dr=0;
- dig_hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
- delay_short(1);
- dig_hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- ucTempData=ucDigStatusTemp08_01;//再先送低8位
- for(i=0;i<8;i++)
- {
- if(ucTempData>=0x80)dig_hc595_ds_dr=1;
- else dig_hc595_ds_dr=0;
- dig_hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
- delay_short(1);
- dig_hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- dig_hc595_st_dr=0;//ST引腳把兩個寄存器的數(shù)據(jù)更新輸出到74HC595的輸出引腳上并且鎖存起來
- delay_short(1);
- dig_hc595_st_dr=1;
- delay_short(1);
- dig_hc595_sh_dr=0; //拉低,抗干擾就增強
- dig_hc595_st_dr=0;
- dig_hc595_ds_dr=0;
- }
- //LED燈的74HC595驅(qū)動函數(shù)
- void hc595_drive(unsigned char ucLedStatusTemp16_09,unsigned char ucLedStatusTemp08_01)
- {
- unsigned char i;
- unsigned char ucTempData;
- hc595_sh_dr=0;
- hc595_st_dr=0;
- ucTempData=ucLedStatusTemp16_09;//先送高8位
- for(i=0;i<8;i++)
- {
- if(ucTempData>=0x80)hc595_ds_dr=1;
- else hc595_ds_dr=0;
- hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
- delay_short(1);
- hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- ucTempData=ucLedStatusTemp08_01;//再先送低8位
- for(i=0;i<8;i++)
- {
- if(ucTempData>=0x80)hc595_ds_dr=1;
- else hc595_ds_dr=0;
- hc595_sh_dr=0; //SH引腳的上升沿把數(shù)據(jù)送入寄存器
- delay_short(1);
- hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- hc595_st_dr=0;//ST引腳把兩個寄存器的數(shù)據(jù)更新輸出到74HC595的輸出引腳上并且鎖存起來
- delay_short(1);
- hc595_st_dr=1;
- delay_short(1);
- hc595_sh_dr=0; //拉低,抗干擾就增強
- hc595_st_dr=0;
- hc595_ds_dr=0;
- }
- void T0_time(void) interrupt 1 //定時中斷
- {
- TF0=0;//清除中斷標志
- TR0=0; //關中斷
- if(ucVoiceLock==0) //原子鎖判斷
- {
- if(uiVoiceCnt!=0)
- {
- uiVoiceCnt--; //每次進入定時中斷都自減1,直到等于零為止。才停止鳴叫
- beep_dr=0;//蜂鳴器是PNP三極管控制,低電平就開始鳴叫。
- }
- else
- {
- ; //此處多加一個空指令,想維持跟if括號語句的數(shù)量對稱,都是兩條指令。不加也可以。
- beep_dr=1;//蜂鳴器是PNP三極管控制,高電平就停止鳴叫。
- }
- }
- if(ucDelayTimerLock==0) //原子鎖判斷
- {
- if(uiDelayTimer>0)
- {
- uiDelayTimer--; //一氣呵成的定時器延時方式的計時器
- }
- }
- if(ucEepromError==1) //EEPROM出錯
- {
- if(ucEepromLock==0)//原子鎖判斷
- {
- uiEepromCnt++;//間歇性蜂鳴器報警的計時器
- }
- }
- key_scan(); //按鍵掃描函數(shù)
- display_drive();//數(shù)碼管字模的驅(qū)動函數(shù)
- TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
- TL0=0x0b;
- TR0=1;//開中斷
- }
- void delay_short(unsigned int uiDelayShort)
- {
- unsigned int i;
- for(i=0;i
- {
- ; //一個分號相當于執(zhí)行一條空語句
- }
- }
- void delay_long(unsigned int uiDelayLong)
- {
- unsigned int i;
- unsigned int j;
- for(i=0;i
- {
- for(j=0;j<500;j++)//內(nèi)嵌循環(huán)的空指令數(shù)量
- {
- ; //一個分號相當于執(zhí)行一條空語句
- }
- }
- }
- void delay_timer(unsigned int uiDelayTimerTemp)
- {
- ucDelayTimerLock=1; //原子鎖加鎖
- uiDelayTimer=uiDelayTimerTemp;
- ucDelayTimerLock=0; //原子鎖解鎖
- /* 注釋一:
- *延時等待,一直等到定時中斷把它減到0為止.這種一氣呵成的定時器方式,
- *可以在延時的時候動態(tài)掃描數(shù)碼管,改善數(shù)碼管的閃爍現(xiàn)象
- */
- while(uiDelayTimer!=0);//一氣呵成的定時器方式延時等待
- }
- void initial_myself(void)//第一區(qū) 初始化單片機
- {
- key_gnd_dr=0; //模擬獨立按鍵的地GND,因此必須一直輸出低電平
- beep_dr=1; //用PNP三極管控制蜂鳴器,輸出高電平時不叫。
- hc595_drive(0x00,0x00);//關閉所有經(jīng)過另外兩個74HC595驅(qū)動的LED燈
- TMOD=0x01;//設置定時器0為工作方式1
- TH0=0xfe; //重裝初始值(65535-500)=65035=0xfe0b
- TL0=0x0b;
- }
- void initial_peripheral(void) //第二區(qū) 初始化外圍
- {
- ucDigDot8=0; //小數(shù)點全部不顯示
- ucDigDot7=0;
- ucDigDot6=0;
- ucDigDot5=0;
- ucDigDot4=0;
- ucDigDot3=0;
- ucDigDot2=0;
- ucDigDot1=0;
- EA=1; //開總中斷
- ET0=1; //允許定時中斷
- TR0=1; //啟動定時中斷
- /* 注釋二:
- * 檢查AT24C02芯片是否存在短路,虛焊,芯片壞了等不工作現(xiàn)象。
- * 在一個特定的地址里把數(shù)據(jù)讀出來,如果發(fā)現(xiàn)不等于0x5a,則重新寫入0x5a,再讀出來
- * 判斷是不是等于0x5a,如果不相等,則芯片有問題,出錯報警提示。
- */
- ucCheckEeprom=read_eeprom(254); //判斷AT24C02是否正常
- if(ucCheckEeprom!=0x5a)//如果不等于特定內(nèi)容。則重新寫入數(shù)據(jù)再判斷一次
- {
- write_eeprom(254,0x5a);//重新寫入標志數(shù)據(jù)
- ucCheckEeprom=read_eeprom(254); //判斷AT24C02是否正常
- if(ucCheckEeprom!=0x5a)//如果還是不等于特定數(shù)字,則芯片不正常
- {
- ucEepromError=1;//表示AT24C02芯片出錯報警
- }
- }
- uiSetData1=read_eeprom_int(0);//讀取uiSetData1,內(nèi)部占用2個字節(jié)地址
- if(uiSetData1>9999) //不在范圍內(nèi)
- {
- uiSetData1=0; //填入一個初始化數(shù)據(jù)
- write_eeprom_int(0,uiSetData1); //存入uiSetData1,內(nèi)部占用2個字節(jié)地址
- }
- uiSetData2=read_eeprom_int(2);//讀取uiSetData2,內(nèi)部占用2個字節(jié)地址
- if(uiSetData2>9999)//不在范圍內(nèi)
- {
- uiSetData2=0;//填入一個初始化數(shù)據(jù)
- write_eeprom_int(2,uiSetData2); //存入uiSetData2,內(nèi)部占用2個字節(jié)地址
- }
- uiSetData3=read_eeprom_int(4);//讀取uiSetData3,內(nèi)部占用2個字節(jié)地址
- if(uiSetData3>9999)//不在范圍內(nèi)
- {
- uiSetData3=0;//填入一個初始化數(shù)據(jù)
- write_eeprom_int(4,uiSetData3); //存入uiSetData3,內(nèi)部占用2個字節(jié)地址
- }
- uiSetData4=read_eeprom_int(6);//讀取uiSetData4,內(nèi)部占用2個字節(jié)地址
- if(uiSetData4>9999)//不在范圍內(nèi)
- {
- uiSetData4=0;//填入一個初始化數(shù)據(jù)
- write_eeprom_int(6,uiSetData4); //存入uiSetData4,內(nèi)部占用2個字節(jié)地址
- }
- }
總結陳詞:
下一節(jié)開始講關于單片機驅(qū)動實時時鐘芯片的內(nèi)容,欲知詳情,請聽下回分解-----利用DS1302做一個實時時鐘。
評論