刚写了一个关于stm32单片机的按键识别的程序。目的,同时识别多个按键,并且不浪费cpu的时间。

关于去抖动,以前以为是在按键的时候,手会抖动。通过程序验证,这个确实是误解。这个应该是防止意外干扰。以我的手按键的速度,单次持续时间小于0.2秒钟。

前提:引脚低电平为按键按下。为每个按键设置3个变量,分别是识别过程的计数器keycnt、识别的结果keystat、可以判定的结果keymod(比如按下、释放、单击、双击、长按等)。

本例 只有单击一种判断,keymod暂时不用。另外考虑,按键判定后,直到松开,不重复判定。具体模式和规则,可根据需要修改。另外要考虑,按键后,要执行的其他动作,是瞬间动作还是长时间动作,比如让灯闪烁,本例没有涉及该步骤。

思路:定时周期内:扫描每个按键的引脚电平,根据预定规则改变对应的计数器;依据计数状态,给出按键结果。

对于具体应用,应进一步人性化。

部分头文件:

// KEY:PB12,PB13,PB14,PB15;PB3,PB4
// KEY On:0 ; KEY Off:1 低电平触发

#ifndef __MINIKEY_

#define __MINIKEY_

#include "stdint.h"
#include "miniled.h"
#include "miniusart.h"
#include "minirtc.h"

#define keynum 6 //On Board KEY NUMBER

#define KEY0 (GPIOB->IDR&0x1000) //PB12
#define KEY1 (GPIOB->IDR&0x2000) //PB13,
#define KEY2 (GPIOB->IDR&0x4000) //PB14
#define KEY3 (GPIOB->IDR&0x8000) //PB15
#define KEY4 (GPIOB->IDR&0x0001) //PB0
#define KEY5 (GPIOB->IDR&0x0002) //PB1
#define KEYON 0
#define KEYOFF 1
#define KEYPRESS 0x02  //单击

#endif

相关功能:

uint8_t keypress=15; // 去抖动参数,用于计数器参考判定按键单击。定时器间隔为0.01秒。
uint8_t keycnt[keynum]; //计数器
uint8_t keystat[keynum]; //0x00/release;0x01/down; 0x02/press;0x04/up; 0x10 /click; 0x20 /double; 0x40 /keep;
uint8_t keymod[keynum]; //定义同keystat,对应bit位置1,则为启动相应功能。

// KEY:PB12,PB13,PB14,PB15

void key_init(void)
{
uint8_t m;
RCC->APB2ENR|=0x08; //enable PORTB clock
GPIOB->CRH&=0x0000FFFF; //clean PB12,PB13,PB14,PB15¡£Per 4 bits for one Port¡£
GPIOB->CRH|=0x88880000; //set PB12,PB13,PB14,PB15
GPIOB->CRL&=0xFFF00000; //clean PB0,PB1¡£PB2,PB3 error¡£
GPIOB->CRL|=0x00008888; //set PB0,PB1¡£PB2,PB3 error¡£
GPIOB->ODR&=0x0FF0;
GPIOB->ODR|=0xF00F;

for(m=0;m<keynum;m++){
keycnt[m]=0;
keystat[m]=0;
keymod[m]=0x10;
}
}

//循环扫描,在定时中执行
void keyscan(void)
{
    uint8_t k;
    for(k=0;k<keynum;k++){
        keyone(k);
    }
}

//识别规则
void keyone(uint8_t nkey)
{
switch(nkey){
case 0:
if(KEY0==KEYON){
    if(keycnt[nkey]==keypress){
     //   keycnt[nkey]=0;//长按视为反复触发按键。
        keystat[nkey]=0x02;
        }else{
        keycnt[nkey]++;
        }
}else{
     keycnt[nkey]=0;
}
break;

…………

}

//后续动作,在定时中执行
void keydo(void)
{
if(keystat[0]==0x02){
keystat[0]=0; //reset key after done
kset0();  //要执行的动作
}
……………………
if(keystat[5]==0x02){
keystat[5]=0; //reset key begin done
led_flush(2);
led_flush(3);
}
}

最新文章

  1. Xilinx DCM 使用---- 输出频率问题
  2. arduino api手册
  3. java中获取比毫秒更为精确的时间
  4. Intellj IDEA快捷键
  5. OO之美
  6. Mysql 5.7.7
  7. How to Build Android Applications Based on FFmpeg by An Example
  8. [MongoDB] Query, update, index and group
  9. 联通光纤上网配置+华为HG8240光猫+TL-WR842N
  10. javascript解析从服务器返回的json格式数据
  11. Solr自动生成ID
  12. JQuery之 serialize() 及serializeArray() 实例介绍
  13. where id in用 order by field 保持排序
  14. BZOJ 1486: [HNOI2009]最小圈 [01分数规划]
  15. QT使用painter绘制文字时的居中显示
  16. Python Skelve 库
  17. linux18.04+jdk11.0.2+hadoop3.1.2部署伪分布式
  18. 物联网智能家居ppt
  19. Redis深入学习笔记(一)Redis启动数据加载流程
  20. Oja’s rule

热门文章

  1. Angular React 和 Vue的比较
  2. 图片上传插件用法,JS语法【三】
  3. 搭建git代码服务器
  4. laravel创建定时任务
  5. 返回变量的类型VarType函数
  6. Mybatis的传参
  7. python自动化Traceback (most recent call last):报错
  8. 转: Executor类
  9. LCOV 如何过滤iostream等系统函数的覆盖率信息
  10. CentOS 下网络报错 Device eth0 does not seem to be present