
简单分析一下按键检测原理,当按键没有按下的时候,单片机的引脚通过上拉电阻R接到VCC,程序在读取这个引脚的电平时值为1(高电平);当按键按下时,该引脚被短接到GND,此时程序读这个引脚的电平时值为0(低电平);因此,按键是否按下可以通过对应引脚的高低电平来反应,我们只要再程序中判断引脚的电平状态就可以识别按键是否按下,从而做出相应的处理;
一切看起来很美好,但现实并非理想状态!
我们在得出上述结论的时候其实忽略了一个重要的问题,那就是现实中按键按下时电压的变化状态不是理想的如下图:

二现实往往是残酷的,由于按键的弹片接触时,并不是一接触就紧紧的闭合,它还存在一定的抖动,尽管这个时间非常的短暂,但是对于执行速度以us为计算单位的微控制器来说就比较漫长了,实际的按键波形是下面这样的:

这样便存在这样一个问题,由于抖动的存在,程序会认为多次按下了按键,这并不是我们所需要的。解决办法就是软件延时消抖,由于抖动的时间大概维持20ms以内,只要软件第一次检测到按键按下时,延时20ms,避开这个抖动区间就可以了;
示例代码:
51 | if ((digitalRead(KEY_UP) == LOW) || (digitalRead(KEY_DOWN) == LOW) || (digitalRead(KEY_LEFT) == LOW) || (digitalRead(KEY_RIGHT) == LOW) || (digitalRead(KEY_ENTER) == LOW)) |
57 | if ((digitalRead(KEY_UP) == LOW) || (digitalRead(KEY_DOWN) == LOW) || (digitalRead(KEY_LEFT) == LOW) || (digitalRead(KEY_RIGHT) == LOW) || (digitalRead(KEY_ENTER) == LOW)) |
60 | if (digitalRead(KEY_UP) == LOW) |
61 | Serial.print( "you press UP\n" ); |
62 | else if (digitalRead(KEY_DOWN) == LOW) |
63 | Serial.print( "you press DOWN\n" ); |
64 | else if (digitalRead(KEY_LEFT) == LOW) |
65 | Serial.print( "you press LEFT\n" ); |
66 | else if (digitalRead(KEY_RIGHT) == LOW) |
67 | Serial.print( "you press RIGHT\n" ); |
68 | else if (digitalRead(KEY_ENTER) == LOW) |
69 | Serial.print( "you press ENTER\n" ); |
程序运行时板载的LED会以1秒为周期闪烁,提示系统正常工作;当按下任意一个按键时,串口监视器会显示当前按下的键位,例如“you press UP”。现象如下图:

源代码下载:
joysticks.zip