標題: 51單片機按鍵消抖方式總結(jié) [打印本頁]

作者: xinzhi1992    時間: 2021-8-20 16:39
標題: 51單片機按鍵消抖方式總結(jié)
電路圖如下:
MCU采用89C52單片機,晶振12MHZ。


1、沒有消除抖動的原始代碼:
  1. #include <REGX52.H>
  2. #include <intrins.h>
  3. sbit KeyValue=P3^7;
  4. unsigned char code segment[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};
  5. //定義數(shù)碼管顯示0~9
  6. void main(){
  7.         static char count=1;
  8.         P2=segment[0]; //開始運行顯示0
  9.         while(1){
  10.                 if(KeyValue==0){
  11.                         P2=segment[count];
  12.                         count++;
  13.                         if(count>=10){   //超過0~9,數(shù)碼管顯示回到0
  14.                                 count=0;
  15.                         }
  16.                 }
  17.         }
  18. }

復(fù)制代碼
2、延時消除抖動

存在如下缺點:

3、使用定時器消抖
原理說明:1次按下+1次抬起構(gòu)成一個按鍵動作,當同時檢測到這兩個動作時,才完成一次按鍵操作。按下時,將按鍵值存儲為0;抬起時,將按鍵值存儲為1。在前一次的按鍵值為0的前提下,檢測當前按鍵值是否為1,如果為1,表示此次按鍵有效,否則此次按鍵無效。


缺點:會占用一個定時
  1. #include <REGX52.H>
  2. #include <intrins.h>
  3. sbit KeyValue=P3^7;
  4. bit  KeyStatus=1;
  5. unsigned char code segment[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};

  6. void main(){
  7.         bit KeySave=1;
  8.         unsigned char count=0;
  9.         P2=segment[0];
  10.         /**************開啟中斷**************************/
  11.         EA=1;
  12.         TMOD=0x01;
  13.         TH0=0xF8;
  14.         TL0=0xCD;
  15.         ET0=1;
  16.         TR0=1;
  17.         while(1){
  18.                 if(KeyStatus!=KeySave){//檢測按鍵值是否改變,初始時按鍵值為1,在此檢測按鍵值是否變?yōu)?,為0則繼續(xù)
  19.                         if(KeySave==0){//如果前一次的按鍵值為0,說明本次按鍵抬起,本次按鍵有效;否則為按鍵按下操作,跳轉(zhuǎn)到最后一步,將按鍵值取反
  20.                                 count++;//對按鍵值+1
  21.                                 if (count>=10){
  22.                                         count=0;
  23.                                 }
  24.                                 P2=segment[count];
  25.                         }
  26.                         KeySave=~KeySave;
  27.                 }
  28.         }
  29. }
  30. void InterruptTimer0() interrupt 1 {
  31.         static unsigned KeyBuff=0xff;
  32.         TH0=0xF8;
  33.         TL0=0xCD;
  34.         KeyBuff=(KeyBuff<<1)|KeyValue;
  35.         switch(KeyBuff){
  36.           case 0xff:
  37.                         KeyStatus=1;
  38.                   break;
  39.           case 0x00:
  40.                         KeyStatus=0;
  41.                   break;
  42.           default:
  43.                   break;
  44.   }        
  45. }
復(fù)制代碼


作者: op5726170    時間: 2021-8-25 20:05
還可以搞兩個for嵌套循環(huán)! 最外層記錄按了多少次,用來處理多次連續(xù)按鍵。內(nèi)層在規(guī)定時間內(nèi)有按鍵動作記為真,超過規(guī)定時間記為假,這樣可以高效消抖
作者: 炒菜    時間: 2021-9-9 09:51
我記的網(wǎng)上有人利用標志位進行消抖,用延時的話有點困難
作者: herui2128    時間: 2021-9-9 14:47
一般來說都是延時5MS,小程序上用不了多少
作者: dyx811    時間: 2021-9-11 19:27
原理分析到位!多謝分享
作者: 我會想你的    時間: 2021-9-30 10:42
其實這個就是狀態(tài)機
作者: DZY11    時間: 2021-10-4 15:45
挺好的啊,我自己也是這樣使用的
作者: Su丿nice    時間: 2021-10-5 07:10
這樣很難嗎?               
                if(K5 == 0)        
                {
                        DelayM(20); q = ~q;                //標志位取反
                        if(q){kai_numl = kai_numl | 1;}
                                     else { kai_numl = kai_numl&0;}
                         if(q){BLK = BLK | 1;}
                                else {BLK= BLK & 0;}
                        while(K5 == 0);
                                
                }
作者: mxdkey    時間: 2021-10-9 17:04
主意不錯,站位留存一下!
作者: 一giao我里giao    時間: 2021-10-11 16:55
if(!key)
while(1)
{
//如果有顯示程序?qū)懸槐樵谶@里
if(key)
  {
  //按鍵要實現(xiàn)的功能
  break;
  }
}
這樣寫只能按一次實現(xiàn)一次





歡迎光臨 (http://www.torrancerestoration.com/bbs/) Powered by Discuz! X3.1