目錄
1、項目需求分析 2
11、項目目標意義 2
12、功能需求分析 2
13、系統開發環境、工具需求分析與選擇 2
14、非功能性需求分析 2
2、項目硬件系統結構 2
21、 系統方案原理圖 2
22、AT89C51原理介紹 2
23 、光電開關模塊原理 3
24、報警模塊原理 4
25、按鍵模塊原理 4
26、數碼管顯示模塊原理 5
27、存儲模塊原理 5
28、傳送帶電機驅動模塊原理 6
3、系統軟件體系架構 6
31 項目軟件系統總架構圖 6
32 顯示子模塊流程圖 9
33 重要的數據結構、參數和函數分析 12
4、項目運行效果展示 13
5、總結與心得體會 14
6、附錄 15
1、項目需求分析 1.1、項目目標意義 工業生產中,很多領域需要自動統計產品的數量,基于光電開關和AT89C51單片機開發的傳送帶計數器完美解決了傳送帶上面運送的產品需要人工計數的煩惱,通過該計數器的應用,企業無論是計數效率和準確度方面都有了質的提升而普及自動計數器也具有非常實際的意義,受到廣大企業的青睞。 1.2、功能需求分析 通過光電開關檢測傳送帶傳送過來的產品,每過一個產品,單片機計數加1,由此實現自動計數功能;設置蜂鳴器在出現異常時可實現報警功能;添加數碼管實現計數的顯示功能;通過EEPROM存儲器,實現計數的存儲功能;添加按鍵實現相關設置功能。 1.3、系統開發環境、工具需求分析與選擇 本系統是基于Keil C51的單片機開發環境進行的開發,Keil C51是美國Keil Software公司出品的51系列兼容單片機C語言軟件開發系統,與匯編相比,C語言在功能上、結構性、可讀性、可維護性上有明顯的優勢,因而易學易用,基于此我們選擇C語言進行編程。 1.4、非功能性需求分析 在電子技術飛速發展的今天,電子產品的智能化和自動化的發展已越來越成熟,每個行業都在尋求自動化來代替人工,基于光電開關和AT89C51單片機開發的傳送帶計數器使用簡單,操作方便,有效節省了人工成本,單位時間內提高了工作效率,給企業帶來更大的收益。 2、項目硬件系統結構 2.1、系統方案原理圖
如圖是整個系統實現的框圖,從圖中我們可以看到整個系統包括單片機模塊、按鍵輸入模塊、計數模塊、存儲模塊、數碼管顯示模塊、傳送帶電機模塊、蜂鳴器和指示燈報警模塊。整個系統通過這幾個模塊協同工作實現傳送帶計數器的功能。
2.2、AT89C51原理介紹 AT89C51是一種帶4K字節閃爍可編程可擦除只讀存儲器(FPEROM—Falsh Programmable and Erasable Read Only Memory)的低電壓,高性能CMOS8位微處理器,俗稱單片機。該器件采用ATMEL高密度非易失存儲器制造技術制造,與工業標準的MCS-51指令集和輸出管腳相兼容。由于將多功能8位CPU和閃爍存儲器組合在單個芯片中,ATMEL的AT89C51是一種高效微控制器,為很多嵌入式控制系統提供了一種靈活性高且價廉的方案。 本系統應用到的AT89C51的I/O輸入輸出功能:在32個I/O口中選擇,實現按鍵、顯示、驅動等功能,I/O口分為P0、P1和P3三大類;P0可以作普通IO口,也可以在讀寫外部存儲器作低8位的地址總線和8位的數據總線;P1口只作為一般IO口;P2作為一般IO口還在在讀寫外部存儲器作高8位的地址總線;P3口除作為一般IO口還為第二輸入/輸出功能(P3.0 串行數據接入端;P3.1 串行數據發送端;P3.2外中斷0輸入端 ;P3.3 外中斷1輸入端;P3.4 定時或計數器TO 的外部輸入端;P3.5 定時或計數器T1的外部輸入端;P3.6 外部數據存儲器寫選通信號;P3.7 外部數據存儲器讀選通信號;) 本系統應用到的定時器功能:實現定時中斷功能,AT89C51有T0和T1兩個定時/計數器,分別有定時和計數兩種模式、4種(T1為3種)工作方式,方式0-方式3,方式0下、計數工作方式時,計數范圍是1-8192,定時工作方式時,定時時間的計算公式是:(2^13-計數初值)*機器周期;方式1下,與方式0的差別僅在于計數位數不同,方式1為16位計數,作為定時方式使用時,定時時間的計算公式是:(2^16-計數初值)*機器周期,計數范圍是1-65536;方式2下,計數滿后自動裝入計數初值,精確定時并簡化定時初值;方式3只適用于T0,T1不能工作在方式3。 2.3 、光電開關模塊原理 光電開關即光電傳感器,是光電接近開關的簡稱,它主要是利用被檢測物對光束的遮擋或反射,由同步回路選通電路,從而檢測物體有無的。光電開關是傳感器的一種,它把發射端和接收端之間光的強弱變化轉化為電流的變化以達到探測的目的。例當光電開關被產品遮擋時,輸出引腳變低電平,單片機檢測到低電平計數一次代表一個產品,當光電開關無遮擋是,輸出引腳變高電平,單片機檢測到高電平不予計數,代表無產品通過。我們這里使用一個開關模擬他的信號,這個開關接入單片機的P3.4口,當按鍵按下一次表示檢測到一個商品。 2.4、報警模塊原理 本系統采用蜂鳴器和LED指示燈做報警元件,蜂鳴器一端接電源,另一端通過三級管接到單片機I/O口進行控制,當需要報警時,單片機控制蜂鳴器的I/O口置高,三極管導通,將蜂鳴器另一端拉低,蜂鳴器導通并鳴響,實現報警。另外LED串聯一個限流電阻,一端連接電源,一端連接單片機的P1.1引腳,單片機控制LED亮滅進行指示作用。 2.5、按鍵模塊原理 本系統采用獨立按鍵和矩陣按鍵結合的方式,獨立按鍵直接用I/O口線構成的單個按鍵電路,其特點式每個按鍵單獨占用一根I/O口線,每個按鍵的工作不會影響其他I/O口線的狀態。獨立式按鍵電路配置靈活,軟件結構簡單,但每個按鍵必須占用一個I/O口線,因此,在按鍵較多時,I/O口線浪費較大,不宜采用。獨立按鍵的軟件常采用查詢式結構。先逐位查詢沒跟I/O口線的輸入狀態,如某一根I/O口線輸入為低電平,則可確認該I/O口線所對應的按鍵已按下,然后,再轉向該鍵的功能處理程序。矩陣按鍵是逐行掃描,然后判斷有沒有按鍵按下,有按下的時候再判斷列,最終確定按鍵的數值。
2.6、數碼管顯示模塊原理 數碼管也稱LED數碼管,晶美、光電、不同行業人士對數碼管的稱呼不一樣,其實都是同樣的產品。數碼管按段數可分為七段數碼管和八段數碼管,八段數碼管比七段數碼管多一個發光二極管單元(多一個小數點顯示);按能顯示多少個“8”可分為1位、2位、3位、4位、5位、6位、7位等數碼管;按發光二極管單元連接方式可分為共陽極數碼管和共陰極數碼管。共陽數碼管是指將所有發光二極管的陽極接到一起形成公共陽極(COM)的數碼管,共陽數碼管在應用時應將公共極COM接到+5V,當某一字段發光二極管的陰極為低電平時,相應字段就點亮,當某一字段的陰極為高電平時,相應字段就不亮。共陰數碼管是指將所有發光二極管的陰極接到一起形成公共陰極(COM)的數碼管,共陰數碼管在應用時應將公共極COM接到地線GND上,當某一字段發光二極管的陽極為高電平時,相應字段就點亮,當某一字段的陽極為低電平時,相應字段就不亮,本系統采用共陽數碼管。 2.7、存儲模塊原理 本系統采用AT24C02存儲芯片,AT24C02芯片是以IIC接口的EEPROM器件。所謂EEPROM即電可擦除可編程只讀存儲器,是ROM的一種。它是只讀存儲器,即掉電可繼續存儲數據,而同時又可以在高于普通電壓的作用下擦除和重寫,這就方便了單片機對其的開發,現在電腦上的ROM很多都是用的EEPROM。 2.8、傳送帶電機驅動模塊原理 本系統采用單片機通過三極管控制繼電器驅動直流電機,根據傳送帶直流電機的功率等參數選用相應的繼電器,當單片機控制電機的I/O口為低時,三極管導通,繼電器吸合,電機電源端導通,電機運轉傳送,當單片機控制電機的I/O口為高時,三極管截止,繼電器斷開,電機電源端斷開,電機停止運轉。 3、系統軟件體系架構 3.1 項目軟件系統總架構圖 整個系統的實現流程圖如圖,從圖中我們可以看到程序首先對外設進行了初始化,然后讀取存儲在芯片中的設定數值,接著初始化定時器之后就進入了主循環。主循環首先讀取了按鍵,然后根據按鍵進行設定和啟停等操作,同時判斷計數數值是否大于等于設置數值,如果到了就報警提示。最后根據不同的模式顯示不同的數值。
/******************************************************************** * 名稱 : Main() * 功能 : 主函數 * 輸入 : 無 * 輸出 : 無 ***********************************************************************/ void Main(void) { uchar press_sure_num=0; uchar Key_num=0; uchar temp=0;
moto=0; //關閉電機 BUZZ=1; //關閉報警 Read_set_num(); //讀取設定的數值 Time0_init(); // 定時器初始化
while(1) { Key_num=KEY_Scan(); //掃描按鍵
if(set_mode==0) //非設定模式 { if(Key_num==12) //啟動 { counter_buf[0]=0;counter_buf[1]=0;counter_buf[2]=0;counter_buf[3]=0; //清除顯示 counter_num=0; //計數清零 moto=1; //打開電機 LED=1; BUZZ=1; //關閉報警 EA=1; //打開計數 } else if(Key_num==13)//停止 { moto=0; //關閉電機 LED=1; BUZZ=1;//關閉報警 EA=0; //關閉計數 } if(Key_num==14) //設置 { set_mode=1; //設置模式置一 set_position=1; moto=0; //關閉電機 EA=0; //關閉計數
} } else//設定模式 { if(Key_num==11)//按下了確定鍵 { set_mode=0;//退出設定 set_position=0; counter_buf[0]=0;counter_buf[1]=0;counter_buf[2]=0;counter_buf[3]=0; //set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值 Write_set_num(); } else if(Key_num<10)//顯示的設定值 {
dis_data_buf[4-set_position]=Key_num; set_position++; if(set_position>4)set_position=4; } else if(Key_num==10)//清除設定 { set_position=1; dis_data_buf[0]=0; dis_data_buf[1]=0; dis_data_buf[2]=0; dis_data_buf[3]=0; } }
set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值
if(counter_num>=set_num) //如果計數數值大于等于設定的數值就停止 { moto=0; //關閉電機 BUZZ=0; //打開報警 LED=0;//
} if(set_mode==0) //正常模式下顯示計數值 { diplay(counter_buf); } else //設定模式下顯示設定值 { diplay(dis_data_buf); }
} }
3.2 顯示子模塊流程圖 這里我們介紹的是顯示子函數,子函數首先是關閉數碼管的顯示,這一步稱為消引。然后把顯示的數據送給其中一個數碼管,最后點亮這個數碼管,等到下一次輪尋就再調用下一個數碼管。
//數碼管的顯示函數 void diplay(uchar *dis_p) { static uchar temp_num=0; static uint flash_time=0; //全部關閉消引 seg_1=0; seg_2=0; seg_3=0; seg_4=0; SEG_DATA=SEG_Table[*(dis_p+temp_num)]; //把數據發送給數碼管的數據接口 switch(temp_num) //動態一次掃描各個數碼管 { case 0: if(set_position==4) { if(flash_time<50) //如果社說的時間小于50那么就打開心事 {seg_1=1;seg_2=0;seg_3=0;seg_4=0;} else if(flash_time<100)//小于100就關閉顯示 {seg_1=0;seg_2=0;seg_3=0;seg_4=0;} else //大于等于100就重新開始閃爍 {flash_time=0;} } else { seg_1=1; seg_2=0; seg_3=0; seg_4=0; } break; case 1: if(set_position==3) { if( flash_time<50) {seg_1=0;seg_2=1;seg_3=0;seg_4=0;} else if(flash_time<100) {seg_1=0;seg_2=0;seg_3=0;seg_4=0;} else {flash_time=0;} } else { seg_1=0; seg_2=1; seg_3=0; seg_4=0; } break; case 2: if(set_position==2) { if( flash_time<50) {seg_1=0;seg_2=0;seg_3=1;seg_4=0;} else if(flash_time<100) {seg_1=0;seg_2=0;seg_3=0;seg_4=0;} else {flash_time=0;} } else { seg_1=0; seg_2=0; seg_3=1; seg_4=0; } break; case 3: if(set_position==1) { if( flash_time<50) {seg_1=0;seg_2=0;seg_3=0;seg_4=1;} else if(flash_time<100) {seg_1=0;seg_2=0;seg_3=0;seg_4=0;} else {flash_time=0;} } else { seg_1=0; seg_2=0; seg_3=0; seg_4=1; } break; } Delay_1ms(2);
temp_num++; if(temp_num>3) temp_num=0;//循環四次
flash_time++; if(set_position==0)flash_time=0; //閃爍
} 3.3 重要的數據結構、參數和函數分析 //讀取設定的目標數值 void Read_set_num(void) { uchar i=0; for(i=0;i<4;i++) { dis_data_buf[ i]=read_rom(i); [ i] if(dis_data_buf[ i]>9)break; } if(i==4) { set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值 } else { for(i=0;i<4;i++) { dis_data_buf[ i]=0; write_rom(i,dis_data_buf[ i]); } } } 這里我們分析一下讀取設定目標數值的子函數,這個子函數在程序的一開始調用一次,主要是讀取上一次存儲的數據,防止其掉電丟失。首選讀24C02得到不同地址內的數據,然后根據存入的順序把數據拼接成設定值給后面使用。 4、項目運行效果展示 這是初始狀態沒有進行操作的時候,如圖 然后我們點擊啟動按鈕,可以看到電機開始轉動了,如圖 我們按下模擬光電的開關,實際數值增加到設定值,此時電機停止并報警,如圖 5、總結與心得體會 這個設計過程中,我們通過在原有的計數器系統進行了改進,使之增添了設置、啟動、停止等的三個控制功能,并添加了報警、顯示、存儲等功能,使之成為一個更加適用,功能更加完備的屬于自己的一個系統。設計結果能夠符合題意,成功完成了此次設計要求,這個過程中,我們花費了大量的時間和精力,更重要的是,我們在學會創新的基礎上,同時還懂得合作精神的重要性,學會了與他人合作。我們掌握的僅僅是理論知識,如何去鍛煉我們的實踐能力?如何把我們所學的專業基礎課理論知識運用到實踐中去呢?我想做類似設計就為我們提供了良好的實踐平臺。
在這次設計中,我們運用到了以前所學的專業知識,如:C語言、模擬和數字電路知識等。雖然過去從未獨立應用過它們,但在學習的過程中帶著問題去學效率很高,這是我做這次設計的又一收獲。 基于Keil C51的單片機開發環境應用范圍廣泛,功能強大,但由于自己應用不夠熟悉,出現了很多不規范的情況,使得在調試過程中遇到很多困難,這是自己軟件開發能力不足的表現,有待提高。本系統通過按鍵進行設定計數功能,需要人為現場近距離操作,可考慮通過遙控設定,改進方案后,可遠距離操作,更好的節省人力。
6、附錄 
單片機源程序如下: - //包含頭文件
- #include<reg52.h>
- #include<intrins.h>
-
- //宏定義方便使用
- #define uint unsigned int
- #define uchar unsigned char
- #define AT24C02 255
- #define AT24C16 2047
- #define AT24CXX AT24C02 //此處修改器件類型24c02或者24c256
-
- sbit SCL = P1^6; //24Cxx接線
- sbit SDA = P1^7;
-
-
- //數碼管位選接口
- sbit seg_1=P3^0;
- sbit seg_2=P3^1;
- sbit seg_3=P3^2;
- sbit seg_4=P3^3;
-
- #define SEG_DATA P0 //數碼管的數據接口
-
-
- //獨立按鍵
- sbit K1=P3^7;
- sbit K2=P3^6;
- sbit K3=P3^5;
-
- //矩陣按鍵的接口
- #define K_PORT P2
-
- //蜂鳴器引腳
- sbit BUZZ=P1^0;
- //LED引腳
- sbit LED=P1^1;
-
- //電機的控機制口
- sbit moto=P1^2;
-
-
- uchar code SEG_Table[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0x88,0x83,0xc6,0xa1,0x86,0x8e};//CA共陽
- char dis_data_buf[4]={0,0,0,1};
- uchar counter_buf[4]={0,0,0,0};
- unsigned long int set_num=0;
- unsigned long int counter_num=0;
- uchar set_position=0;
- uchar set_mode=0;
-
-
- void Delay_1ms(uint i)//1ms延時
- {
- uchar x,j;
- for(j=0;j<i;j++)
- for(x=0;x<=148;x++);
- }
-
- void delay_us(uint i)
- {
- uint t;
- for(t=0;t<i;t++)
- _nop_();
-
- }
-
-
-
- /*開始I2C數據發送或接收*/
- void start_rom()//在時鐘線為高,數據線從高電平跳到低電平時,開始傳送
- {
- SDA=1; //數據先
- delay_us(2);
- SCL=1; //時鐘線為高,等數據線下降才開始
- delay_us(5);
- SDA=0; //數據線下降拉,啟動了
- delay_us(5);
- SCL=0;
- }
-
- /*結束數據傳送*/
- void stop_rom()
- {
- SDA=0; //時鐘線為高時,數據線上升沿為結束信號
- delay_us(2);
- SCL=1;
- delay_us(4);
- SDA=1;
- delay_us(4);
- SCL=0;
- }
-
- //************************************************************************//
- //函數名稱:void ack_rom()
- //函數功能:ROM應答信號
- //**********************************************************************//
- void ack_rom()
- {
- SDA=1;
- delay_us(2);
- SCL=0;
- delay_us(2);
- SCL=1;
- delay_us(2);
- SCL=0;
- delay_us(2);
- }
-
- //***********************************************************************//
- //函數名稱:void no_ack_rom()
- //函數功能:無需應答
- //***********************************************************************//
- void no_ack_rom()
- {
- SDA=1;
- delay_us(2);
- SCL=1;
- delay_us(2);
- SCL=0;
- delay_us(2);
- }
-
- //**********************************************//
- //函數名稱:void write_byte_rom(uchar data)
- //函數功能:向ROM 寫入字節
- //傳入參數:data 待寫入的字節
- //返回參數:無
- //**********************************************//
- void write_byte_rom(uchar dataa)
- {
- uchar i=8;
- SDA = 0;
- SCL=0;
- _nop_();_nop_();_nop_();_nop_();_nop_();
- while(i--)//寫數據循環,從高位開始
- {
- if((dataa & 0x80)) SDA=1; //將IO 口拉高,寫入1
- else SDA=0; //將IO 口拉低,寫入0
- delay_us(5);
- dataa<<=1;
- SCL=1;
- delay_us(5);
- SCL=0;
- delay_us(5);
- }
- }
-
- uchar read_byte_rom()
- {
- uchar i,k;
- for(i=0;i<8;i++)
- {
- SCL=1;
- delay_us(5);
- k=(k<<1)|SDA;
- SCL=0;
- delay_us(5);
- }
- return k;
- }
-
- uchar read_rom(uint add)
- {
- uchar da;
- start_rom();
- write_byte_rom(0xa0);
- ack_rom();
- if(AT24CXX>AT24C16)
- {
- //write_byte_rom(addr/256); //寫入地址高八位
- write_byte_rom(add>>8); //寫入地址高八位
- ack_rom();
- write_byte_rom(add%256); //寫入地址低八位
- //write_byte_rom(addr&0xff); //寫入地址低八位
- }
- else
- write_byte_rom(add);
-
- ack_rom();
- start_rom();
- write_byte_rom(0xa1);
- ack_rom();
- da=read_byte_rom();
- no_ack_rom();
- stop_rom();
- return da;
-
- }
-
- //**********************************************//
- //函數名稱:void write_rom(uchar addr,uchar data)
- //函數功能:寫數據到ROM
- //傳入參數:addr 寫入的地址
- //返回參數:read_data 待寫入的數據
- //**********************************************//
- void write_rom(uint addr,uchar dataa)
- {
- start_rom();
- write_byte_rom(0xa0); //選擇寫操作
- ack_rom();
- if(AT24CXX>AT24C16)
- {
- //write_byte_rom(addr/256); //寫入地址高八位
- write_byte_rom(addr>>8); //寫入地址高八位
- ack_rom();
- write_byte_rom(addr%256); //寫入地址低八位
- //write_byte_rom(addr&0xff); //寫入地址低八位
- }
- else
- write_byte_rom(addr);
- ack_rom();
- write_byte_rom(dataa); //寫入數據
-
- ack_rom();
- stop_rom();
-
- }
-
-
-
- //*********************************
- //按鍵掃描程序
- //*********************************
- uchar KEY_Scan()
- {
- uchar table[3]={0xB0,0xd0,0xE0};
- uchar temp=0,temp_data=0,temp_key_num=0XFF;
- uchar i=0,k=0;
- static uchar key_up=0;
-
- K_PORT = 0xF0;
- if(key_up==0)
- {
-
- for(k=0;k<4;k++)
- {
- temp_data=~(1<<k); //逐行拉低判斷有沒有按鍵按下
- _nop_();
- K_PORT=temp_data;
- if(K_PORT!=temp_data)//有按鍵按下
- {
- Delay_1ms(10);//去抖動
- if(K_PORT!=temp_data)//有按鍵按下
- {
- key_up=1;
- temp=K_PORT&0xf0; //取高字節,即列
- for(i=0;i<3;i++)
- {
- if(temp==table[i])
- {
- break;
- }
- }
- if(i!=3)//有效按鍵
- {
- temp_key_num=i+k*3; //獲取按鍵
- break;
- }
-
- }
- }
- }
-
- //判斷獨立按鍵
- if((K1==0||K2==0||K3==0))
- {
- key_up=1;
- Delay_1ms(10);//去抖動
- if(K1==0)temp_key_num=12;
- else if(K2==0)temp_key_num=13;
- else if(K3==0)temp_key_num=14;
- }
-
- }
- else
- {
- if((K_PORT==0xf0)&&(K1==1)&&(K2==1)&&(K3==1))key_up=0; //松開按鍵
- temp_key_num=0XFF;
- }
- return temp_key_num;// 返回鍵值
- }
-
- //數碼管的顯示函數
- void diplay(uchar *dis_p)
- {
- static uchar temp_num=0;
- static uint flash_time=0;
- //全部關閉消引
- seg_1=0;
- seg_2=0;
- seg_3=0;
- seg_4=0;
- SEG_DATA=SEG_Table[*(dis_p+temp_num)]; //把數據發送給數碼管的數據接口
- switch(temp_num) //動態一次掃描各個數碼管
- {
- case 0:
- if(set_position==4)
- {
- if(flash_time<50) //如果社說的時間小于50那么就打開心事
- {seg_1=1;seg_2=0;seg_3=0;seg_4=0;}
- else if(flash_time<100)//小于100就關閉顯示
- {seg_1=0;seg_2=0;seg_3=0;seg_4=0;}
- else //大于等于100就重新開始閃爍
- {flash_time=0;}
- }
- else
- {
- seg_1=1;
- seg_2=0;
- seg_3=0;
- seg_4=0;
- }
- break;
- case 1:
- if(set_position==3)
- {
- if( flash_time<50)
- {seg_1=0;seg_2=1;seg_3=0;seg_4=0;}
- else if(flash_time<100)
- {seg_1=0;seg_2=0;seg_3=0;seg_4=0;}
- else
- {flash_time=0;}
- }
- else
- {
- seg_1=0;
- seg_2=1;
- seg_3=0;
- seg_4=0;
- }
- break;
- case 2:
- if(set_position==2)
- {
- if( flash_time<50)
- {seg_1=0;seg_2=0;seg_3=1;seg_4=0;}
- else if(flash_time<100)
- {seg_1=0;seg_2=0;seg_3=0;seg_4=0;}
- else
- {flash_time=0;}
- }
- else
- {
- seg_1=0;
- seg_2=0;
- seg_3=1;
- seg_4=0;
- }
- break;
- case 3:
- if(set_position==1)
- {
- if( flash_time<50)
- {seg_1=0;seg_2=0;seg_3=0;seg_4=1;}
- else if(flash_time<100)
- {seg_1=0;seg_2=0;seg_3=0;seg_4=0;}
- else
- {flash_time=0;}
- }
- else
- {
- seg_1=0;
- seg_2=0;
- seg_3=0;
- seg_4=1;
- }
- break;
- }
- Delay_1ms(2);
-
- temp_num++;
- if(temp_num>3) temp_num=0;//循環四次
-
- flash_time++;
- if(set_position==0)flash_time=0; //閃爍
-
-
- }
- //計數器0的初始化
- void Time0_init(void)
- {
- TMOD=0x05;
- TH0=0xff;
- TL0=0xff;
- ET0=1;
- TR0=1;
- EA=1;
-
- }
-
- //讀取設定的目標數值
- void Read_set_num(void)
- {
- uchar i=0;
- for(i=0;i<4;i++)
- {
- dis_data_buf[i]=read_rom(i);
- if(dis_data_buf[i]>9)break;
- }
- if(i==4)
- {
- set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值
- }
- else
- {
- for(i=0;i<4;i++)
- {
- dis_data_buf[i]=0;
- write_rom(i,dis_data_buf[i]);
- }
- }
- }
- //向eeprom中寫入存儲數據
- void Write_set_num()
- {
- uchar i=0;
- for(i=0;i<4;i++)
- write_rom(i,dis_data_buf[i]);
-
-
- }
- /********************************************************************
- * 名稱 : Main()
- * 功能 : 主函數
- * 輸入 : 無
- * 輸出 : 無
- ***********************************************************************/
- void Main(void)
- {
- uchar press_sure_num=0;
- uchar Key_num=0;
- uchar temp=0;
-
- moto=0; //關閉電機
- BUZZ=1; //關閉報警
- Read_set_num(); //讀取設定的數值
- Time0_init(); // 定時器初始化
-
- while(1)
- {
- Key_num=KEY_Scan(); //掃描按鍵
-
- if(set_mode==0) //非設定模式
- {
- if(Key_num==12) //啟動
- {
- counter_buf[0]=0;counter_buf[1]=0;counter_buf[2]=0;counter_buf[3]=0; //清除顯示
- counter_num=0; //計數清零
- moto=1; //打開電機
- LED=1;
- BUZZ=1; //關閉報警
- EA=1; //打開計數
- }
- else if(Key_num==13)//停止
- {
- moto=0; //關閉電機
- LED=1;
- BUZZ=1;//關閉報警
- EA=0; //關閉計數
- }
- if(Key_num==14) //設置
- {
- set_mode=1; //設置模式置一
- set_position=1;
- moto=0; //關閉電機
- EA=0; //關閉計數
-
- }
-
-
- }
- else//設定模式
- {
- if(Key_num==11)//按下了確定鍵
- {
- set_mode=0;//退出設定
- set_position=0;
- counter_buf[0]=0;counter_buf[1]=0;counter_buf[2]=0;counter_buf[3]=0;
- //set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值
- Write_set_num();
- }
- else if(Key_num<10)//顯示的設定值
- {
-
- dis_data_buf[4-set_position]=Key_num;
- set_position++;
- if(set_position>4)set_position=4;
- }
- else if(Key_num==10)//清除設定
- {
- set_position=1;
- dis_data_buf[0]=0;
- dis_data_buf[1]=0;
- dis_data_buf[2]=0;
- dis_data_buf[3]=0;
- }
- }
-
- set_num=(dis_data_buf[0]*1000)+(dis_data_buf[1]*100)+(dis_data_buf[2]*10)+dis_data_buf[3]; //計算出計數的數值
-
- if(counter_num>=set_num) //如果計數數值大于等于設定的數值就停止
- {
- moto=0; //關閉電機
- BUZZ=0; //打開報警
- LED=0;//
-
- }
- if(set_mode==0) //正常模式下顯示計數值
- {
- diplay(counter_buf);
- }
- else //設定模式下顯示設定值
- {
- diplay(dis_data_buf);
- }
-
-
- }
- }
-
- //計數器0中斷
- void Time0_counter() interrupt 1
- {
- TH0=0xff;
- TL0=0xff;
-
- if(TF0==0)
- {
- if(counter_num>9999)counter_num=0; //最大計數到9999,然后從零開始
- else
- {
- counter_num++; //計數加一
- counter_buf[3]++;
- if(counter_buf[3]>9) //個位大于9十位就加一
- {
- counter_buf[3]=0;
- counter_buf[2]++;
- }
- if(counter_buf[2]>9) //十位大于9百位就加一
- {
- counter_buf[2]=0;
- counter_buf[1]++;
- }
- if(counter_buf[1]>9)
- {
- counter_buf[1]=0;
- counter_buf[0]++;
- }
- if(counter_buf[0]>9)
- {
- counter_buf[0]=0;
- }
- }
- }
- }
復制代碼
|