找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

帖子
查看: 25447|回復(fù): 15
打印 上一主題 下一主題
收起左側(cè)

單片機(jī)按鍵消抖方式詳解

  [復(fù)制鏈接]
跳轉(zhuǎn)到指定樓層
樓主
ID:205966 發(fā)表于 2017-5-29 23:57 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式

通常按鍵所用的開關(guān)都是機(jī)械彈性開關(guān),當(dāng)機(jī)械觸點斷開、閉合時,由于機(jī)械觸點的彈性作用,一個按鍵開關(guān)在閉合時不會馬上就穩(wěn)定的接通,在斷開時也不會一下子徹底斷開,而是在閉合和斷開的瞬間伴隨了一連串的抖動,如圖8-10


所示。
圖8-10按鍵抖動狀態(tài)圖
按鍵穩(wěn)定閉合時間長短是由操作人員決定的,通常都會在100ms以上,刻意快速按的話能達(dá)到40-50ms左右,很難再低了。抖動時間是由按鍵的機(jī)械特性決定的,一般都會在10ms以內(nèi),為了確保程序?qū)Π存I的一次閉合或者一次斷開只響應(yīng)一次,必須進(jìn)行按鍵的消抖處理。當(dāng)檢測到按鍵狀態(tài)變化時,不是立即去響應(yīng)動作,而是先等待閉合或斷開穩(wěn)定后再進(jìn)行處理。按鍵消抖可分為硬件消抖和軟件消抖。
硬件消抖:
利用電容的充放電特性來對抖動過程中產(chǎn)生的電壓毛刺進(jìn)行平滑處理,從而實現(xiàn)消抖。但實際應(yīng)用中,這種方式的效果往往不是很好,而且還增加了成本和電路復(fù)雜度,所以實際中使用的并不多。如圖8-11所示:
1/7頁


圖8-11硬件電容消抖
軟件實現(xiàn)消抖:
最簡單的消抖原理,就是當(dāng)檢測到按鍵狀態(tài)變化后,先等待一個10ms左右的延時時間,讓抖動消失后再進(jìn)行一次按鍵狀態(tài)檢測,如果與剛才檢測到的狀態(tài)相同,就可以確認(rèn)按鍵已經(jīng)穩(wěn)定的動作了。
程序如下:
#include<reg52.h>
sbitADDR0=P1^0;
sbitADDR1=P1^1;
sbitADDR2=P1^2;
sbitADDR3=P1^3;
sbitENLED=P1^4;
sbitKEY1=P2^4;
sbitKEY2=P2^5;
sbitKEY3=P2^6;
sbitKEY4=P2^7;
unsignedcharcodeLedChar[]={//數(shù)碼管顯示字符轉(zhuǎn)換表
0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,
2/7頁
0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E
};
voiddelay();
voidmain(){
bitkeybuf=1;//按鍵值暫存,臨時保存按鍵的掃描值
bitbackup=1;//按鍵值備份,保存前一次的掃描值
unsignedcharcnt=0;//按鍵計數(shù),記錄按鍵按下的次數(shù)
ENLED=0;//選擇數(shù)碼管DS1進(jìn)行顯示
ADDR3=1;
ADDR2=0;
ADDR1=0;
ADDR0=0;
P2=0xF7;//P2.3置0,即KeyOut1輸出低電平
//顯示按鍵次數(shù)初值P0=LedChar[cnt];
while(1){
keybuf=KEY4;//把當(dāng)前掃描值暫存
//當(dāng)前值與前次值不相等說明此時按鍵有動作if(keybuf!=backup){
delay();//延時大約10ms
//判斷掃描值有沒有發(fā)生改變,即按鍵抖動
//如果前次值為0,則說明當(dāng)前是彈起動作if(keybuf==KEY4){if(backup==0){
cnt++;//按鍵次數(shù)+1
//只用1個數(shù)碼管顯示,所以加到10就清零重新開始
if(cnt>=10){
cnt=0;
}
P0=LedChar[cnt];//計數(shù)值顯示到數(shù)碼管上
}
backup=keybuf;//更新備份為當(dāng)前值,以備進(jìn)行下次比較
}
}
}
3/7頁
}
/*軟件延時函數(shù),延時約10ms*/
voiddelay(){
unsignedinti=1000;
while(i--);
}
這個程序用了一個簡單的算法實現(xiàn)了按鍵的消抖。作為這種很簡單的演示程序,我們可以這樣來寫,但是實際做項目開發(fā)的時候,程序量往往很大,各種狀態(tài)值也很多,while(1)這個主循環(huán)要不停的掃描各種狀態(tài)值是否有發(fā)生變化,及時的進(jìn)行任務(wù)調(diào)度,如果程序中間加了這種delay延時操作后,很可能某一事件發(fā)生了,但是我們程序還在進(jìn)行delay延時操作中,當(dāng)這個事件發(fā)生完了,程序還在delay操作中,當(dāng)我們delay完事再去檢查的時候,已經(jīng)晚了,已經(jīng)檢測不到那個事件了。
為了避免這種情況的發(fā)生,我們要盡量縮短while(1)循環(huán)一次所用的時間,而需要進(jìn)行長時間延時的操作,必須想其它的辦法來處理。
那么消抖操作所需要的延時該怎么處理呢?
舉個例子:我們啟用一個定時中斷,每2ms進(jìn)一次中斷,掃描一次按鍵狀態(tài)并且存儲起來,連續(xù)掃描8次后,看看這連續(xù)8次的按鍵狀態(tài)是否是一致的。8次按鍵的時間大概是16ms,這16ms內(nèi)如果按鍵狀態(tài)一直保持一致,那就可以確定現(xiàn)在按鍵處于穩(wěn)定的階段,而非處于抖動的階段,如圖


8-12。
圖8-12按鍵連續(xù)掃描判斷
假如左邊時間是起始0時刻,每經(jīng)過2ms左移一次,每移動一次,判斷當(dāng)前連續(xù)的
4/7頁
8次按鍵狀態(tài)是不是全1或者全0,如果是全1則判定為彈起,如果是全0則判定為按下,如果0和1交錯,就認(rèn)為是抖動,不做任何判定。想一下,這樣是不是比簡單的延時更加可靠?
利用這種方法,就可以避免通過延時消抖占用單片機(jī)執(zhí)行時間,而是轉(zhuǎn)化成了一種按鍵狀態(tài)判定而非按鍵過程判定,我們只對當(dāng)前按鍵的連續(xù)16ms的8次狀態(tài)進(jìn)行判斷,而不再關(guān)心它在這16ms內(nèi)都做了什么事情,那么下面就按照這種思路用程序?qū)崿F(xiàn)出來,同樣只以K4為例。
#include<reg52.h>
sbitADDR0=P1^0;
sbitADDR1=P1^1;
sbitADDR2=P1^2;
sbitADDR3=P1^3;
sbitENLED=P1^4;
sbitKEY1=P2^4;
sbitKEY2=P2^5;
sbitKEY3=P2^6;
sbitKEY4=P2^7;
unsignedcharcodeLedChar[]={//數(shù)碼管顯示字符轉(zhuǎn)換表
0xC0,0xF9,0xA4,0xB0,0x99,0x92,0x82,0xF8,
0x80,0x90,0x88,0x83,0xC6,0xA1,0x86,0x8E
};
bitKeySta=1;//當(dāng)前按鍵狀態(tài)
voidmain(){
bitbackup=1;//按鍵值備份,保存前一次的掃描值
//按鍵計數(shù),記錄按鍵按下的次數(shù)unsignedcharcnt=0;
EA=1;//使能總中斷
ENLED=0;//選擇數(shù)碼管DS1進(jìn)行顯示

評分

參與人數(shù) 2黑幣 +6 收起 理由
zzwk123456 + 5 很給力!
sjwss + 1 很給力!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏2 分享淘帖 頂 踩
回復(fù)

使用道具 舉報

沙發(fā)
ID:193314 發(fā)表于 2017-5-30 22:59 | 只看該作者
為什么我們不站在巨人的肩膀上來思考問題呢。對這個典型問題要自己去yy呢。
回復(fù)

使用道具 舉報

板凳
ID:198323 發(fā)表于 2017-6-3 15:45 | 只看該作者
圖在哪兒呢?
回復(fù)

使用道具 舉報

地板
ID:206560 發(fā)表于 2017-6-9 04:47 | 只看該作者
按鍵防抖并不復(fù)雜,但是這段程序把問題復(fù)雜化了,而且吧各種應(yīng)用問題混淆了。
按鍵防抖思路: 如果檢測到按下,延時約20ms再次檢測,如果還是按下,可確認(rèn)為按下,否則認(rèn)為是抖動。
keyloop: if key=0 { xc=0;
                           while(key!=0)    xc++;
                         }
            if (xc<MAX)  goto keyloop;
MAX 就是按下多長時間認(rèn)為是真的按下的最大計數(shù)值,自己根據(jù)經(jīng)驗和實際情況設(shè)定。 xc是計數(shù)器。=1時,按鍵按下并放開了�?衫^續(xù)按鍵處理,否則, 表示這是一次抖動,可繼續(xù)判斷按鍵按下。
回復(fù)

使用道具 舉報

5#
ID:215848 發(fā)表于 2017-6-29 15:58 | 只看該作者
我感覺利用定時器去計時來做延時不會去比較大的影響查詢模式里別的功能,是一種不錯的去抖方式。
回復(fù)

使用道具 舉報

6#
ID:215848 發(fā)表于 2017-6-29 15:59 | 只看該作者
wangminfu 發(fā)表于 2017-6-9 04:47
按鍵防抖并不復(fù)雜,但是這段程序把問題復(fù)雜化了,而且吧各種應(yīng)用問題混淆了。
按鍵防抖思路: 如果檢測到 ...

程序里再加上while循環(huán)感覺不太好,會影響別的邏輯繼續(xù)運(yùn)行。
回復(fù)

使用道具 舉報

7#
ID:137736 發(fā)表于 2017-7-6 23:23 | 只看該作者
定時器加按鍵標(biāo)志消抖比較靠譜。
回復(fù)

使用道具 舉報

8#
ID:221290 發(fā)表于 2017-7-21 10:52 | 只看該作者
很有用的!
回復(fù)

使用道具 舉報

9#
ID:223262 發(fā)表于 2017-7-29 17:42 | 只看該作者
定時器加按鍵標(biāo)志消抖比較靠譜。
回復(fù)

使用道具 舉報

10#
ID:230395 發(fā)表于 2017-9-1 12:32 | 只看該作者
挺實用的,就是復(fù)雜化了
回復(fù)

使用道具 舉報

11#
ID:235153 發(fā)表于 2017-9-23 19:21 來自手機(jī) | 只看該作者
我也懂,但是怎樣加到程序里面呢?
回復(fù)

使用道具 舉報

12#
ID:238170 發(fā)表于 2017-10-10 00:58 | 只看該作者
中斷定時不停地檢測按鍵消抖感覺比較簡單,就是得調(diào)。。
回復(fù)

使用道具 舉報

13#
ID:241535 發(fā)表于 2017-10-22 19:17 | 只看該作者
這個很實用啊,一直想知道怎么實現(xiàn)的,給個贊吧
回復(fù)

使用道具 舉報

14#
ID:847776 發(fā)表于 2020-12-5 16:25 | 只看該作者
這個只是按下防抖吧,請問松開的時候如何防抖呢?
回復(fù)

使用道具 舉報

15#
ID:856401 發(fā)表于 2020-12-7 22:31 | 只看該作者
這個是金沙灘教程里的例子
回復(fù)

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表