第46節(jié):利用AT24C02進行掉電后的數(shù)據保存
一個AT24C02可以存儲256個字節(jié),地址范圍是(0至255)。利用AT24C02存儲數(shù)據時,要教會大家六個知識點:
第一個:單片機操作AT24C02的通訊過程也就是IIC的通訊過程, IIC通訊過程是一個要求一氣呵成的通訊過程,中間不能被其它中斷影響時序出錯,因此在整個通訊過程中應該先關閉總中斷,完成之后再開中斷。
第二個:在寫入或者讀取完一個字節(jié)之后,一定要加上一段延時時間。在11.0592M晶振的系統(tǒng)中,寫入數(shù)據時經驗值用delay_short(2000),讀取數(shù)據時經驗值用delay_short(800)。否則在連續(xù)寫入或者讀取一串數(shù)據時容易丟失數(shù)據。如果一旦發(fā)現(xiàn)丟失數(shù)據,應該適當繼續(xù)把這個時間延長,尤其是在寫入數(shù)據時。
第三個:如何初始化EEPROM數(shù)據的方法。系統(tǒng)第一次上電時,我們從EEPROM讀取出來的數(shù)據有可能超出了范圍,可能是ff。這個時候我們應該給它填入一個初始化的數(shù)據,這一步千萬別漏了。
第四個:在時序中,發(fā)送ACK確認信號時,要記得把數(shù)據線eeprom_sda_dr_s設置為輸入的狀態(tài)。對于51單片機來說,只要把eeprom_sda_dr_s=1就可以。而對于PIC或者AVR單片機來說,它們都是帶方向寄存器的,就不能直接eeprom_sda_dr_s=1,而要直接修改方向寄存器,把它設置為輸入狀態(tài)。在本驅動程序中,我沒有對ACK信號進行出錯判斷,因為我這么多年一直都是這樣用也沒出現(xiàn)過什么問題。
第五個: 提醒各位讀者在硬件上應該注意的問題,單片機跟AT24C02通訊的2根IO口都要加上一個4.7K左右的上拉電阻。凡是在IIC通訊場合,都要加上拉電阻。AT24C02的WP引腳一定要接地,否則存不進數(shù)據。
第六個:舊版的朱兆祺51學習板在硬件上有一個bug,AT24C02的第8個引腳VCC懸空了!!!,讀者記得把它飛線連接到5V電源處。新版的朱兆祺51學習板已經改過來了。
具體內容,請看源代碼講解。
(1)硬件平臺:
基于朱兆祺51單片機學習板。
(2)實現(xiàn)功能:
(1)硬件平臺:
基于朱兆祺51單片機學習板。
(2)實現(xiàn)功能:
4個被更改后的參數(shù)斷電后不丟失,數(shù)據可以保存,斷電再上電后還是上一次最新被修改的數(shù)據。
顯示和獨立按鍵部分根據第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)切換不同的窗口。
顯示和獨立按鍵部分根據第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 //按鍵去抖動延時的時間
- void initial_myself(void);
- void initial_peripheral(void);
- void delay_short(unsigned int uiDelayShort);
- void delay_long(unsigned int uiDelaylong);
- //驅動數(shù)碼管的74HC595
- void dig_hc595_drive(unsigned char ucDigStatusTemp16_09,unsigned char ucDigStatusTemp08_01);
- void display_drive(void); //顯示數(shù)碼管字模的驅動函數(shù)
- void display_service(void); //顯示的窗口菜單服務程序
- //驅動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ù)據
- void write_eeprom(unsigned int address,unsigned char dd); //往一個地址存入一個字節(jié)數(shù)據
- unsigned int read_eeprom_int(unsigned int address); //從一個地址讀取出一個int類型的數(shù)據
- void write_eeprom_int(unsigned int address,unsigned int uiWriteData); //往一個地址存入一個int類型的數(shù)據
- void T0_time(void);//定時中斷函數(shù)
- void key_service(void); //按鍵服務的應用程序
- void key_scan(void);//按鍵掃描函數(shù) 放在定時中斷里
- 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; //蜂鳴器的驅動IO口
- sbit eeprom_scl_dr=P3^7; //時鐘線
- sbit eeprom_sda_dr_sr=P3^6; //數(shù)據的輸出線和輸入線
- 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ù)碼管要顯示的內容
- unsigned char ucDigShow7;//第7位數(shù)碼管要顯示的內容
- unsigned char ucDigShow6;//第6位數(shù)碼管要顯示的內容
- unsigned char ucDigShow5;//第5位數(shù)碼管要顯示的內容
- unsigned char ucDigShow4;//第4位數(shù)碼管要顯示的內容
- unsigned char ucDigShow3;//第3位數(shù)碼管要顯示的內容
- unsigned char ucDigShow2;//第2位數(shù)碼管要顯示的內容
- unsigned char ucDigShow1;//第1位數(shù)碼管要顯示的內容
- 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;//中間過渡變量
- //根據原理圖得出的共陰數(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(); //顯示的窗口菜單服務程序
- }
- }
- //AT24C02驅動程序
- 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ù)據之前要先置一,表示數(shù)據輸入
- eeprom_scl_dr=1;
- delay_short(15);
- eeprom_scl_dr=0;
- delay_short(15);
- //在本驅動程序中,我沒有對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ù)據之前要先置一,表示數(shù)據輸入
- 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ù)據之前要先置一,表示數(shù)據輸入
- 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ù)據
- {
- unsigned char dd,cAddress;
- cAddress=address; //把低字節(jié)地址傳遞給一個字節(jié)變量。
- /* 注釋一:
- * IIC通訊過程是一個要求一氣呵成的通訊過程,中間不能被其它中斷影響時序出錯,因此
- * 在整個通訊過程中應該先關閉總中斷,完成之后再開中斷。但是,這樣就會引起另外一個新
- * 問題,如果關閉總中斷的時間太長,會導致動態(tài)數(shù)碼管不能及時均勻的掃描,在操作EEPROM時,
- * 數(shù)碼管就會出現(xiàn)閃爍的現(xiàn)象,解決這個問題最好的辦法就是在做項目中盡量不要用動態(tài)掃描數(shù)碼管
- * 的方案,應該用靜態(tài)顯示的方案。那么程序上還有沒有改善的方法?有的,下一節(jié)我會講這個問題
- * 的改善方法。
- */
- EA=0; //禁止中斷
- start24(); //IIC通訊開始
- write24(0xA0); //此字節(jié)包含讀寫指令和芯片地址兩方面的內容。
- //指令為寫指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- write24(cAddress); //發(fā)送讀取的存儲地址(范圍是0至255)
- ack24(); //發(fā)送應答信號
- start24(); //開始
- write24(0xA1); //此字節(jié)包含讀寫指令和芯片地址兩方面的內容。
- //指令為讀指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- dd=read24(); //讀取一個字節(jié)
- ack24(); //發(fā)送應答信號
- stop24();//停止
- /* 注釋二:
- * 在寫入或者讀取完一個字節(jié)之后,一定要加上一段延時時間。在11.0592M晶振的系統(tǒng)中,
- * 寫入數(shù)據時經驗值用delay_short(2000),讀取數(shù)據時經驗值用delay_short(800)。
- * 否則在連續(xù)寫入或者讀取一串數(shù)據時容易丟失數(shù)據。如果一旦發(fā)現(xiàn)丟失數(shù)據,
- * 應該適當繼續(xù)把這個時間延長,尤其是在寫入數(shù)據時。
- */
- delay_short(800);//此處最關鍵,此處的延時時間一定要,而且要足夠長,此處也是導致動態(tài)數(shù)碼管閃爍的根本原因
- EA=1; //允許中斷
- return(dd);
- }
- void write_eeprom(unsigned int address,unsigned char dd) //往一個地址存入一個字節(jié)數(shù)據
- {
- unsigned char cAddress;
- cAddress=address; //把低字節(jié)地址傳遞給一個字節(jié)變量。
- EA=0; //禁止中斷
- start24(); //IIC通訊開始
- write24(0xA0); //此字節(jié)包含讀寫指令和芯片地址兩方面的內容。
- //指令為寫指令。地址為"000"的信息,此信息由A0,A1,A2的引腳決定
- ack24(); //發(fā)送應答信號
- write24(cAddress); //發(fā)送寫入的存儲地址(范圍是0至255)
- ack24(); //發(fā)送應答信號
- write24(dd);//寫入存儲的數(shù)據
- ack24(); //發(fā)送應答信號
- stop24();//停止
- delay_short(2000);//此處最關鍵,此處的延時時間一定要,而且要足夠長,此處也是導致動態(tài)數(shù)碼管閃爍的根本原因
- EA=1; //允許中斷
- }
- unsigned int read_eeprom_int(unsigned int address) //從一個地址讀取出一個int類型的數(shù)據
- {
- unsigned char ucReadDataH;
- unsigned char ucReadDataL;
- unsigned intuiReadDate;
- ucReadDataH=read_eeprom(address); //讀取高字節(jié)
- ucReadDataL=read_eeprom(address+1);//讀取低字節(jié)
- uiReadDate=ucReadDataH;//把兩個字節(jié)合并成一個int類型數(shù)據
- uiReadDate=uiReadDate<<8;
- uiReadDate=uiReadDate+ucReadDataL;
- return uiReadDate;
- }
- void write_eeprom_int(unsigned int address,unsigned int uiWriteData) //往一個地址存入一個int類型的數(shù)據
- {
- 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ù)據
- 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ù)據
- ucTemp4=uiSetData1/1000;
- ucTemp3=uiSetData1%1000/100;
- ucTemp2=uiSetData1%100/10;
- ucTemp1=uiSetData1%10;
- //再過渡需要顯示的數(shù)據到緩沖變量里,讓過渡的時間越短越好
- if(uiSetData1<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內容
- }
- if(uiSetData1<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內容
- }
- if(uiSetData1<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內容
- }
- break;
- case 2://顯示P--2窗口的數(shù)據
- 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ù)據
- ucTemp3=uiSetData2%1000/100;
- ucTemp2=uiSetData2%100/10;
- ucTemp1=uiSetData2%10;
- if(uiSetData2<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內容
- }
- if(uiSetData2<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內容
- }
- if(uiSetData2<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內容
- }
- break;
- case 3://顯示P--3窗口的數(shù)據
- 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ù)據
- ucTemp3=uiSetData3%1000/100;
- ucTemp2=uiSetData3%100/10;
- ucTemp1=uiSetData3%10;
- if(uiSetData3<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內容
- }
- if(uiSetData3<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內容
- }
- if(uiSetData3<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內容
- }
- break;
- case 4://顯示P--4窗口的數(shù)據
- 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ù)據
- ucTemp3=uiSetData4%1000/100;
- ucTemp2=uiSetData4%100/10;
- ucTemp1=uiSetData4%10;
- if(uiSetData4<1000)
- {
- ucDigShow4=10;//如果小于1000,千位顯示無
- }
- else
- {
- ucDigShow4=ucTemp4;//第4位數(shù)碼管要顯示的內容
- }
- if(uiSetData4<100)
- {
- ucDigShow3=10;//如果小于100,百位顯示無
- }
- else
- {
- ucDigShow3=ucTemp3;//第3位數(shù)碼管要顯示的內容
- }
- if(uiSetData4<10)
- {
- ucDigShow2=10;//如果小于10,十位顯示無
- }
- else
- {
- ucDigShow2=ucTemp2;//第2位數(shù)碼管要顯示的內容
- }
- ucDigShow1=ucTemp1;//第1位數(shù)碼管要顯示的內容
- }
- 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 由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd1Update=1;//窗口1更新顯示
- break;
- case 2:
- uiSetData2++;
- if(uiSetData2>9999) //最大值是9999
- {
- uiSetData2=9999;
- }
- write_eeprom_int(2,uiSetData2); //存入EEPROM,由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd2Update=1;//窗口2更新顯示
- break;
- case 3:
- uiSetData3++;
- if(uiSetData3>9999) //最大值是9999
- {
- uiSetData3=9999;
- }
- write_eeprom_int(4,uiSetData3); //存入EEPROM,由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd3Update=1;//窗口3更新顯示
- break;
- case 4:
- uiSetData4++;
- if(uiSetData4>9999) //最大值是9999
- {
- uiSetData4=9999;
- }
- write_eeprom_int(6,uiSetData4); //存入EEPROM,由于內部有延時函數(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,由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd1Update=1;//窗口1更新顯示
- break;
- case 2:
- uiSetData2--;
- if(uiSetData2>9999)
- {
- uiSetData2=0;//最小值是0
- }
- write_eeprom_int(2,uiSetData2); //存入EEPROM,由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd2Update=1;//窗口2更新顯示
- break;
- case 3:
- uiSetData3--;
- if(uiSetData3>9999)
- {
- uiSetData3=0;//最小值是0
- }
- write_eeprom_int(4,uiSetData3); //存入EEPROM,由于內部有延時函數(shù),所以此處會引起數(shù)碼管閃爍
- ucWd3Update=1;//窗口3更新顯示
- break;
- case 4:
- uiSetData4--;
- if(uiSetData4>9999)
- {
- uiSetData4=0;//最小值是0
- }
- write_eeprom_int(6,uiSetData4); //存入EEPROM,由于內部有延時函數(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驅動函數(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ù)據送入寄存器
- 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ù)據送入寄存器
- delay_short(1);
- dig_hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- dig_hc595_st_dr=0;//ST引腳把兩個寄存器的數(shù)據更新輸出到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驅動函數(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ù)據送入寄存器
- 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ù)據送入寄存器
- delay_short(1);
- hc595_sh_dr=1;
- delay_short(1);
- ucTempData=ucTempData<<1;
- }
- hc595_st_dr=0;//ST引腳把兩個寄存器的數(shù)據更新輸出到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; //關中斷
- /* 注釋三:
- * 此處多增加一個原子鎖,作為中斷與主函數(shù)共享數(shù)據的保護,實際上是借鑒了"紅金龍吸味"關于原子鎖的建議.
- */
- if(ucVoiceLock==0) //原子鎖判斷
- {
- if(uiVoiceCnt!=0)
- {
- uiVoiceCnt--; //每次進入定時中斷都自減1,直到等于零為止。才停止鳴叫
- beep_dr=0;//蜂鳴器是PNP三極管控制,低電平就開始鳴叫。
- }
- else
- {
- ; //此處多加一個空指令,想維持跟if括號語句的數(shù)量對稱,都是兩條指令。不加也可以。
- beep_dr=1;//蜂鳴器是PNP三極管控制,高電平就停止鳴叫。
- }
- }
- key_scan(); //按鍵掃描函數(shù)
- display_drive();//數(shù)碼管字模的驅動函數(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++)//內嵌循環(huán)的空指令數(shù)量
- {
- ; //一個分號相當于執(zhí)行一條空語句
- }
- }
- }
- void initial_myself(void)//第一區(qū) 初始化單片機
- {
- /* 注釋四:
- * 矩陣鍵盤也可以做獨立按鍵,前提是把某一根公共輸出線輸出低電平,
- * 模擬獨立按鍵的觸發(fā)地,本程序中,把key_gnd_dr輸出低電平。
- * 朱兆祺51學習板的S1就是本程序中用到的一個獨立按鍵。
- */
- key_gnd_dr=0; //模擬獨立按鍵的地GND,因此必須一直輸出低電平
- beep_dr=1; //用PNP三極管控制蜂鳴器,輸出高電平時不叫。
- hc595_drive(0x00,0x00);//關閉所有經過另外兩個74HC595驅動的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; //啟動定時中斷
- /* 注釋五:
- * 如何初始化EEPROM數(shù)據的方法。在使用EEPROM時,這一步初始化很關鍵!
- * 第一次上電時,我們從EEPROM讀取出來的數(shù)據有可能超出了范圍,可能是ff。
- * 這個時候我們應該給它填入一個初始化的數(shù)據,這一步千萬別漏了。另外,
- * 由于int類型數(shù)據占用2個字節(jié),所以以下4個數(shù)據挨著的地址是0,2,4,6.
- */
- uiSetData1=read_eeprom_int(0);//讀取uiSetData1,內部占用2個字節(jié)地址
- if(uiSetData1>9999) //不在范圍內
- {
- uiSetData1=0; //填入一個初始化數(shù)據
- write_eeprom_int(0,uiSetData1); //存入uiSetData1,內部占用2個字節(jié)地址
- }
- uiSetData2=read_eeprom_int(2);//讀取uiSetData2,內部占用2個字節(jié)地址
- if(uiSetData2>9999)//不在范圍內
- {
- uiSetData2=0;//填入一個初始化數(shù)據
- write_eeprom_int(2,uiSetData2); //存入uiSetData2,內部占用2個字節(jié)地址
- }
- uiSetData3=read_eeprom_int(4);//讀取uiSetData3,內部占用2個字節(jié)地址
- if(uiSetData3>9999)//不在范圍內
- {
- uiSetData3=0;//填入一個初始化數(shù)據
- write_eeprom_int(4,uiSetData3); //存入uiSetData3,內部占用2個字節(jié)地址
- }
- uiSetData4=read_eeprom_int(6);//讀取uiSetData4,內部占用2個字節(jié)地址
- if(uiSetData4>9999)//不在范圍內
- {
- uiSetData4=0;//填入一個初始化數(shù)據
- write_eeprom_int(6,uiSetData4); //存入uiSetData4,內部占用2個字節(jié)地址
- }
- }
總結陳詞:
IIC通訊過程是一個要求一氣呵成的通訊過程,中間不能被其它中斷影響時序出錯,因此,在整個通訊過程中應該先關閉總中斷,完成之后再開中斷。但是,這樣就會引起另外一個新問題,如果關閉總中斷的時間太長,會導致動態(tài)數(shù)碼管不能及時均勻的掃描,在按鍵更改參數(shù),內部操作EEPROM時,數(shù)碼管就會出現(xiàn)短暫明顯的閃爍現(xiàn)象,解決這個問題最好的辦法就是在做項目中盡量不要用動態(tài)掃描數(shù)碼管的方案,應該用靜態(tài)顯示的方案。那么在程序上還有沒有改善這種現(xiàn)象的方法?當然有。欲知詳情,請聽下回分解-----操作AT24C02時,利用“一氣呵成的定時器方式”改善數(shù)碼管的閃爍現(xiàn)象。
評論