微信搜索ReCclay,也可免费阅读博主蓝桥系列所有文章,后台回复“代码”即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题、免费下载CSDN资源等多项福利,还在等什么呢?快快扫码关注,学习才不会迷路

这里再向各位同学推荐一个CSDN博主 ReRrain 的蓝桥备赛博客,博主秉持初学者思路,向你讲述自己蓝桥备赛的心路历程,娓娓道来蓝桥备赛经验,个人觉得非常不错,值得细细品读。


文章目录

  • 一、基础理论
    • 1.1、循环查询
    • 1.2、定时查询
    • 1.3、中断响应
  • 二、动手实验
    • 2.1、独立按键
    • 2.2、矩阵键盘
    • 2.3、长按键
  • 三、总结

导读:《蓝桥杯单片机组》专栏文章是博主2018年参加蓝桥杯的单片机组比赛所做的学习笔记,在当年的比赛中,博主是获得了省赛一等奖,国赛二等奖的成绩。成绩虽谈不上最好,但至少问心无愧。如今2021年回头再看该系列文章,仍然感触颇多。为了能更好地帮助到单片机初学者,今年特地抽出时间对当年的文章逻辑和结构进行重构,以达到初学者快速上手的目的。需要指出的是,由于本人水平有限,如有错误还请读者指出,非常感谢。那么,接下来让我们一起开始愉快的学习吧。

不积跬步无以至千里,不积小流无以成江海。


通过前面几节的学习,相信对CT107D的外设驱动套路都已经熟悉的差不多了,今天我们来继续学习更难一点的外设:按键。这里再附一个博主当时学习金沙滩51单片机时,所写的一篇理解按键扫描思想的博文:https://blog.csdn.net/ReCclay/article/details/75453278。建议搭配本文一起使用,这样理解起来会更方便!

一、基础理论

按键涉及到的重要知识点就是扫描和消抖了!

关于扫描,主要三种循环查询,定时查询,中断响应,当然各有优缺点,这里来总结下先。

1.1、循环查询

思想:在一个循环函数里不断地扫描按键值,获取按下的按键。

优点:实现简单。
缺点:消抖需要浪费宝贵的CPU时间,且实时性不足(等待)。

1.2、定时查询

思想:在中断服务函数里扫描按键“活”的按键值,根据按键按下的值然后存入缓冲区,等主函数有需要时,再来处理按键消息。(关于消息机制其实是一个很有意思的东西,这里这样称不知道准不准确。)

优点:避免消抖浪费时间,不会丢失捕捉按键按下,容易实现按键按下,长按,以及弹起等动作的识别。
缺点:需要使用定时器中断,要知道单片机最宝贵的资源莫过于定时器。

1.3、中断响应

思想:按键按下触发中断,获取相应的按键值,需要进行消抖处理。

优点:实时性好。
缺点:需要微控制器支持中断,并且消抖浪费CPU资源。

通过上面的分析我们也不难猜出,其实应用比较好的还是定时查询的方式,既可以识别多种按键状态,还不必消抖浪费CPU资源。

二、动手实验

2.1、独立按键

1-2短接实现矩阵按键;2-3短接实现独立按键。

1个独立按键是每2ms扫描一次(进一次中断保存一下当前值),获取连续8个当前值,也就是耗费 2*8 = 16ms。

程序功能:独立按键,同时使用一个数码管实现按一下+1的操作。(注意J5插针在右边)

/*
*******************************************************************************
* 文件名:
* 描  述:
* 作  者:CLAY
* 版本号:v1.0.0
* 日  期:
* 备  注:S4每次加1,S5每次加2,S6每次加3,S7每次加4
*
*******************************************************************************
*/#include <stc15.h>sbit KEY_IN_1 = P3^3;
sbit KEY_IN_2 = P3^2;
sbit KEY_IN_3 = P3^1;
sbit KEY_IN_4 = P3^0;typedef unsigned char u8;
typedef unsigned int  u16;
typedef unsigned long u32;u8 code LedChar[] = {  0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
u8 LedBuff[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};
u8 KeySta[4] = {1, 1, 1, 1};
u8 KeyCodeMap[4] = {'1', '2', '3', '4'};u8 T0RH;
u8 T0RL;
u16 cnt = 0;void CloseFucker();
void ConfigTimer0(u16 ms);
void ShowNumber(u16 dat);
void KeyDriver();void main()
{CloseFucker();ConfigTimer0(2);//2ms一扫。EA = 1;ShowNumber(0);while(1){  KeyDriver();}
}void KeyAction(u8 keycode)
{if(keycode == '1'){cnt += 1;ShowNumber(cnt);} else if(keycode == '2'){cnt += 2;ShowNumber(cnt);} else if(keycode == '3'){cnt += 3;ShowNumber(cnt);} else if(keycode == '4'){cnt += 4;ShowNumber(cnt);}
}void KeyDriver()
{u8 i;static u8 backup[4] = {1, 1, 1, 1};for(i=0; i<4; i++){if(KeySta[i] != backup[i]){if(backup[i] != 0){KeyAction(KeyCodeMap[i]);}backup[i] = KeySta[i];}}
}void CloseFucker()
{P2 = (P2 & 0x1F) | 0x80;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xA0;P0 = 0xAF;P2 = P2 & 0x1F;
}void ConfigTimer0(u16 ms)
{u32 tmp;tmp = 11059200 / 12;tmp = (tmp * ms) / 1000;tmp = 65536 - tmp;T0RH = (u8)(tmp >> 8);T0RL = (u8)tmp;TMOD &= 0xF0;TMOD |= 0x01;TH0 = T0RH;TL0 = T0RL;ET0 = 1;TR0 = 1;
}void ShowNumber(u16 dat)
{char i;u8 buf[8];for(i=0; i<8; i++){buf[i] = dat % 10;dat /= 10;}for(i=7; i>0; i--){if(buf[i] == 0)LedBuff[i] = 0xFF;elsebreak;}for( ; i>=0; i--){LedBuff[i] = LedChar[buf[i]];}
}void LedScan()
{static u8 index = 0;P2 = (P2 & 0x1F) | 0xE0;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xC0;P0 = 0x80 >> index;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xE0;P0 = LedBuff[index];P2 = P2 & 0x1F;if(index < 7)index++;elseindex = 0;
}void KeyScan()
{u8 i;static u8 keybuff[4] = {0xFF, 0xFF, 0xFF, 0xFF};keybuff[0] = (keybuff[0] << 1) | KEY_IN_1;keybuff[1] = (keybuff[1] << 1) | KEY_IN_2;keybuff[2] = (keybuff[2] << 1) | KEY_IN_3;keybuff[3] = (keybuff[3] << 1) | KEY_IN_4;for(i=0; i<4; i++){if(keybuff[i] == 0xFF){KeySta[i] = 1;}else if(keybuff[i] == 0x00){KeySta[i] = 0;}else{}}
}   void interruptTimer0() interrupt 1
{TH0 = T0RH;TL0 = T0RL;   LedScan();KeyScan();
}

2.2、矩阵键盘

1个独立按键的时候,一端是接地的。

同理,矩阵按键无非就是软件设置分别接地而已。

关于扫描时间,这里如果还是2ms, 8个扫描值的话。那么4行按键,每个按键扫8次,也就是248 = 64ms…有点长了。我们改成,1ms一扫,每个按键扫4次。144 = 16ms 和 1个独立按键的时间一样!

矩阵按键映射关系

程序功能:按键对应显示 0-9

/*
*******************************************************************************
* 文件名:
* 描  述:
* 作  者:CLAY
* 版本号:v1.0.0
* 日  期:
* 备  注:显示对应的0-9
*
*******************************************************************************
*/#include <stc15.h>typedef unsigned char u8;
typedef unsigned int  u16;
typedef unsigned long u32;sbit KEY_OUT_1 = P3^0;
sbit KEY_OUT_2 = P3^1;
sbit KEY_OUT_3 = P3^2;
sbit KEY_OUT_4 = P3^3;
sbit KEY_IN_4 = P3^4;
sbit KEY_IN_3 = P3^5;
sbit KEY_IN_2 = P4^2;
sbit KEY_IN_1 = P4^4;u8 code LedChar[] = {0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
u8 LedBuff[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};u8 KeySta[4][4] = {{1, 1, 1, 1}, {1, 1, 1, 1,}, {1, 1, 1, 1}, {1, 1, 1, 1}
};u8 KeyCodeMap[4][4] = {  {'1', '2',  '3',  0x26},{'4', '5',  '6',  0x25},{'7', '8',  '9',  0x28},{'0', 0x1B, 0x0D, 0x27}
};u8 T0RH;
u8 T0RL;void CloseFucker();
void ConfigTimer0(u16 ms);
void KeyDriver();void main()
{CloseFucker();ConfigTimer0(1);EA = 1;while(1){KeyDriver();}
}void ShowNumber(u8 dat)
{char i;u8 buf[8];for(i=0; i<8; i++){buf[i] = dat % 10;dat /= 10;}for(i=7; i>0; i--){if(buf[i] == 0)LedBuff[i] = 0xFF;elsebreak;}for( ; i>=0; i--){LedBuff[i] = LedChar[buf[i]];}
}void KeyAction(u8 keycode)
{if((keycode >= '0') && (keycode <= '9')){ShowNumber(keycode - '0');}
}void KeyDriver()
{u8 i, j;static u8 backup[4][4] = {{1, 1, 1, 1}, {1, 1, 1, 1,}, {1, 1, 1, 1}, {1, 1, 1, 1}};for(i=0; i<4; i++){for(j=0; j<4; j++){if(KeySta[i][j] != backup[i][j]){if(backup[i][j] != 0){KeyAction(KeyCodeMap[i][j]);}backup[i][j] = KeySta[i][j];}}}
}void CloseFucker()
{P2 = (P2 & 0x1F) | 0x80;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xA0;P0 = 0xAF;P2 = P2 & 0x1F;
}void ConfigTimer0(u16 ms)
{u32 tmp;tmp = 11059200 / 12;tmp = (tmp * ms) / 1000;tmp = 65536 - tmp;T0RH = (u8)(tmp >> 8);T0RL = (u8)tmp;TMOD &= 0xF0;TMOD |= 0x01;TH0 = T0RH;TL0 = T0RL;ET0 = 1;TR0 = 1;
}void LedScan()
{static u8 index = 0;P2 = (P2 & 0x1F) | 0xE0;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xC0;P0 = 0x80 >> index;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xE0;P0 = LedBuff[index];P2 = P2 & 0x1F;if(index < 7)index++;elseindex = 0;
}void KeyScan()
{u8 i;static u8 keyout = 0;static u8 keybuff[4][4] = {{0xFF, 0xFF, 0xFF, 0xFF}, {0xFF, 0xFF, 0xFF, 0xFF},{0xFF, 0xFF, 0xFF, 0xFF}, {0xFF, 0xFF, 0xFF, 0xFF}};switch(keyout){case 0: KEY_OUT_1 = 0; KEY_OUT_4 = 1; break;case 1: KEY_OUT_2 = 0; KEY_OUT_1 = 1; break;case 2: KEY_OUT_3 = 0; KEY_OUT_2 = 1; break;case 3: KEY_OUT_4 = 0; KEY_OUT_3 = 1; break;default : break;}keybuff[keyout][0] = (keybuff[keyout][0] << 1) | KEY_IN_1;keybuff[keyout][1] = (keybuff[keyout][1] << 1) | KEY_IN_2;keybuff[keyout][2] = (keybuff[keyout][2] << 1) | KEY_IN_3;keybuff[keyout][3] = (keybuff[keyout][3] << 1) | KEY_IN_4;for(i=0; i<4; i++){if((keybuff[keyout][i] & 0x0F) == 0x0F)KeySta[keyout][i] = 1;else if((keybuff[keyout][i] & 0x0F) == 0x00)KeySta[keyout][i] = 0;else{}}keyout++;keyout &= 0x03;
}void interruptTimer0() interrupt 1
{TH0 = T0RH;TL0 = T0RL;LedScan();KeyScan();}

2.3、长按键

如果上面所介绍的都没有问题了的话,就可以在其上的基础上再来了解一下长按键的实现了! 拿独立按键来说,短按下只加一回,长按一直加,思路也很简单,用到了阈值的思路,这个要特别注意长按的加入不能影响到短按!

设置一个像KeySta的全局变量KeyDownTime,用来保存每个按键按下的时间累加,只要弹起就清零,这个是在KeyScan()里面进行操作的,也是和KeySta状态再一起进行判断的!
然后还需要个TimeThr这个在KeyDriver()里面,初始值为1000。如果检测到按下,执行按键动作函数,继续往下执行,如果检测到某个按键的KeyDownTime不为0,再判断是否大于阈值,大于阈值也要执行按键动作函数,然后让阈值增大,调节阈值增量可以控制增长速度。一旦KeyDownTime等于0,就是按键弹起来了,让阈值回归1000。

看下怎么实现吧!以独立按键的实验为例,矩阵按键同理。

程序功能:S4每次加1,S5每次加2,S6每次加3,S7每次加4

/*
*******************************************************************************
* 文件名:
* 描  述:
* 作  者:CLAY
* 版本号:v1.0.0
* 日  期:
* 备  注:S4每次加1,S5每次加2,S6每次加3,S7每次加4
*
*******************************************************************************
*/#include <stc15.h>sbit KEY_IN_1 = P3^3;
sbit KEY_IN_2 = P3^2;
sbit KEY_IN_3 = P3^1;
sbit KEY_IN_4 = P3^0;typedef unsigned char u8;
typedef unsigned int  u16;
typedef unsigned long u32;u8 code LedChar[] = {  0xC0, 0xF9, 0xA4, 0xB0, 0x99, 0x92, 0x82, 0xF8,0x80, 0x90, 0x88, 0x83, 0xC6, 0xA1, 0x86, 0x8E
};
u8 LedBuff[] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF
};
u8 KeySta[4] = {1, 1, 1, 1};
u16 KeyDownTime[4] = {0, 0, 0, 0};
u8 KeyCodeMap[4] = {'1', '2', '3', '4'};u8 T0RH;
u8 T0RL;
u16 cnt = 0;void CloseFucker();
void ConfigTimer0(u16 ms);
void ShowNumber(u16 dat);
void KeyDriver();void main()
{CloseFucker();ConfigTimer0(2);//2ms一扫。EA = 1;ShowNumber(0);while(1){  KeyDriver();}
}void KeyAction(u8 keycode)
{if(keycode == '1'){cnt += 1;ShowNumber(cnt);} else if(keycode == '2'){cnt += 2;ShowNumber(cnt);} else if(keycode == '3'){cnt += 3;ShowNumber(cnt);} else if(keycode == '4'){cnt += 4;ShowNumber(cnt);}
}void KeyDriver()
{u8 i;static u8 backup[4] = {1, 1, 1, 1};static u16 TimeThr[4] = {1000, 1000, 1000, 1000};for(i=0; i<4; i++){if(KeySta[i] != backup[i]){if(backup[i] != 0){KeyAction(KeyCodeMap[i]);}backup[i] = KeySta[i];}if(KeyDownTime[i] > 0){if(KeyDownTime[i] > TimeThr[i]){KeyAction(KeyCodeMap[i]);TimeThr[i] += 200;}  }else{TimeThr[i] = 1000;}}
}void CloseFucker()
{P2 = (P2 & 0x1F) | 0x80;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xA0;P0 = 0xAF;P2 = P2 & 0x1F;
}void ConfigTimer0(u16 ms)
{u32 tmp;tmp = 11059200 / 12;tmp = (tmp * ms) / 1000;tmp = 65536 - tmp;T0RH = (u8)(tmp >> 8);T0RL = (u8)tmp;TMOD &= 0xF0;TMOD |= 0x01;TH0 = T0RH;TL0 = T0RL;ET0 = 1;TR0 = 1;
}void ShowNumber(u16 dat)
{char i;u8 buf[8];for(i=0; i<8; i++){buf[i] = dat % 10;dat /= 10;}for(i=7; i>0; i--){if(buf[i] == 0)LedBuff[i] = 0xFF;elsebreak;}for( ; i>=0; i--){LedBuff[i] = LedChar[buf[i]];}
}void LedScan()
{static u8 index = 0;P2 = (P2 & 0x1F) | 0xE0;P0 = 0xFF;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xC0;P0 = 0x80 >> index;P2 = P2 & 0x1F;P2 = (P2 & 0x1F) | 0xE0;P0 = LedBuff[index];P2 = P2 & 0x1F;if(index < 7)index++;elseindex = 0;
}void KeyScan()
{u8 i;static u8 keybuff[4] = {0xFF, 0xFF, 0xFF, 0xFF};keybuff[0] = (keybuff[0] << 1) | KEY_IN_1;keybuff[1] = (keybuff[1] << 1) | KEY_IN_2;keybuff[2] = (keybuff[2] << 1) | KEY_IN_3;keybuff[3] = (keybuff[3] << 1) | KEY_IN_4;for(i=0; i<4; i++){if(keybuff[i] == 0xFF){KeySta[i] = 1;KeyDownTime[i] = 0;}else if(keybuff[i] == 0x00){KeySta[i] = 0;KeyDownTime[i] += 4;}else{}}
}   void interruptTimer0() interrupt 1
{TH0 = T0RH;TL0 = T0RL;   LedScan();KeyScan();
}

三、总结

1、要充分利用独立按键和矩阵按键再次感受模块化编程的便利,应用层和底层分离,维护修改记忆都方便,一石好几鸟。

2、注意理解程序中KEY_IN和KEY_OUT引脚定义以及KeyCodeMap的定义。

小结:本篇文章主要介绍了单片机学习中的一个重头戏:按键操作,并结合了常见的按键操作方式:独立按键、矩阵按键和长按键进行了详细的介绍。在该部分学习中比较困难的可能无外乎两点:一个是中断思想的理解,另一个则是按键扫描思想的理解。记住,这是摆在每个初学者面前的两座大山,大家都是这样慢慢啃过来的。好好琢磨,再假以时日相信你一定会融会贯通的。

希望大家多多支持我的原创文章。如有错误,请大家及时指正,非常感谢。


微信搜索ReCclay,即可免费阅读博主蓝桥系列所有文章,后台回复“代码”即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题、免费下载CSDN资源等多项福利,还在等什么呢?快快扫码关注,学习才不会迷路

【蓝桥杯单片机组模块】4、按键模块相关推荐

  1. 【蓝桥杯单片机组模块】14、STC15定时器2的PWM使用

    微信搜索:ReCclay,也可免费阅读博主蓝桥系列所有文章,后台回复"代码"即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题.免费下载CSDN资源等 ...

  2. 蓝桥杯单片机DS18b20单总线测温模块常见问题解决

    蓝桥杯单片机DS18b20单总线测温模块常见问题解决 有道是:"溪水声声留我住,梅花朵朵唤人回" DS18b20测温模块作为一个比较简单.稳定的蓝桥杯单片机外设模块, 使用时却经常 ...

  3. 蓝桥杯单片机组——程序框架及客观题

    文章目录 前言 程序框架 main+中断 两段式代码结构 单片机运行流程 代码风格 客观题 总结 目录 前言 前面两篇主要是介绍了蓝桥省赛的一些参赛技巧,此篇主要是分享程序框架和一些客观题的链接. 程 ...

  4. 【蓝桥杯单片机组实战】2、高级计算器

    微信搜索:ReCclay,也可免费阅读博主蓝桥系列所有文章,后台回复"代码"即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题.免费下载CSDN资源等 ...

  5. 【蓝桥杯单片机组】客观题(赛前必看)

    微信搜索:ReCclay,也可免费阅读博主蓝桥系列所有文章,后台回复"代码"即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题.免费下载CSDN资源等 ...

  6. 模板分享,不知道就亏了——蓝桥杯单片机组做题套路和技巧

    蓝桥杯比赛 单片机组 做题套路和技巧 前言 方法 1.记模块 2.分析框图 3.循序渐进 前言   完成一个完整的题目,需要你对各个模块的熟悉使用以及严密的逻辑思维,然而这还不够,在有限的时间完整的完 ...

  7. 第jiu届蓝桥杯单片机省赛真题_第九届蓝桥杯单片机组省赛试题.pdf

    第九届蓝桥杯单片机组省赛试题 "彩灯控制器"的程序设计与调试 (70 分) 一.基本要求 1.1 使用CT107D 单片机竞赛板,完成"彩灯控制器"功能的程序设 ...

  8. 【蓝桥杯单片机组】备赛实战问题记录

    微信搜索:ReCclay,也可免费阅读博主蓝桥系列所有文章,后台回复"代码"即可获取蓝桥所有备赛代码!关注博主公众号,还可拥有加入博主粉丝群实时沟通技术难题.免费下载CSDN资源等 ...

  9. 十三届蓝桥杯单片机组省赛真题程序解析

    第一次参加蓝桥杯,线上比赛,记录一下 比赛刚开始了十分钟才做上客观题,做上了之后才发现看一次题警告一次,当时就蚌埠住了,随便做了做就交了,手册也没怎么查(查一下直接给了3次黄牌,就没敢查了)其实老师1 ...

最新文章

  1. php swoole process,Swoole_process实现进程池的方法
  2. Python技术分享:ndarray对象的常用属性
  3. HDU-3622 Bomb Game 2sat
  4. DS18B20 驱动编写
  5. 安装EPP的调试Zend Debugger
  6. windows桌面的“我的电脑”“IE”等消失的解决方法
  7. php mysql查询中文乱码_解决php mysql查询插入中文乱码问题_PHP教程
  8. Java常用的工具类库介绍
  9. 传智播客 C/C++学习笔记 const
  10. java代码实现 取放_java大对象存取的简单实现的代码
  11. smali语法基础详解
  12. 1068 万绿丛中一点红 (20 分)
  13. 浅谈数据库用户表结构设计和第三方登录
  14. Python 气象数据分析
  15. CVPR 2021 Oral | Transformer再发力!华南理工和微信提出UP-DETR:无监督预训练检测器...
  16. 如何搭建nginx服务器?
  17. python往数据库批量插入多条数据
  18. 极限保号性和积分保号性
  19. C/C++编程:虚函数与纯虚函数
  20. Linux文件系统笔记

热门文章

  1. 【Python】快速创建一个简易 HTTP 服务器(http.server)
  2. 高薪程序员面试题精讲系列77之如何防止前端页面重复提交请求?
  3. 如何增加技术员的自信心呢?
  4. GCC 编译命令和基本使用
  5. 分布式京东全站爬取分析
  6. 软考网络工程师(八)
  7. 设计模式之工厂模式05
  8. 华为平板服务器响应异常,华为平板触摸屏没反应
  9. RF2O-2D激光里程计算法源码解析与相关公式的详细推导
  10. 31岁北京程序媛拿到国企offer,但是薪资打5折,跳不跳?