矩阵键盘介绍
在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式。
采用逐行或逐列的“扫描”,就可以读出任何位置按键的状态。
结构:在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式。在矩阵式键盘中,每条水平线和垂直线在交叉处不直接连通,而是通过一个按键加以连接。这样,一个端口(如P1口)就可以构成4*4=16个按键,比之直接将端口线用于键盘多出了一倍,而且线数越多,区别越明显,比如再多加一条线就可以构成20键的键盘,而直接用端口线则只能多出一键(⑨键) 由此可见,在需要的键数比较多时,采用矩阵法来做键盘是合理的。
扫描的概念
数码管扫描(输出扫描)
原理:显示第1位→显示第2位→显示第3位→……,然后快速循环这个过程,最终实现所有数码管同时显示的效果。因为它的扫描速度是非常快的,根据人的肉眼现象,你所看到的扫描都是同时进行显示的。这就是少量 IO 口,连接到矩阵减少 IO 口的一个目的。
矩阵键盘扫描(输入扫描)
原理:读取第1行(列)→读取第2行(列) →读取第3行(列) → ……,然后快速循环这个过程,最终实现所有按键同时检测的效果。原理:跟数码管是极其相像的,扫描的过程其实是由电脑的显卡来进行扫描的这个是一个其它的一个知识点。其实对于显示器来说这个扫描的概念是非常广泛的,因为这个像素点是非常多的几乎所有的显示器都会采用矩阵来进行扫描的。
以上两种扫描方式的共性:节省I/O口😶
矩阵键盘原理图
如何用单片机去扫描这个按键从而去获取键码🤔
独立按键和矩阵按键的相同之处
独立按键它是把按键的公共的一端全部连接在了低电平上,然后另一端连接到了 IO 口上。
矩阵按键它是我们把①行④个单独的去拿出来看一下(S1、S2、S3、S4)这一行它的公共端它如果说把它连接到GND(P17~P14)如果不要了的话。会发现这个矩阵键盘其实就是和我们说的独立按键是一模一样的!
扫描矩阵键盘的第一步:如果说是按行那么就把第一个接到GND上,然后用④个if分别进行判断(P13~P10) if(P13==0)那么就证明S1是按下的。同理:if(P12==0)的话那么就是S2是按下的!那么后面两个也是同样的!第一行就可以完美的解决了。
判断第二行的话,我们只需要把第一行给 1,第二行给 0,第三行给 1,第四行给 1。就可以了。因为如果给 1 的话,你的另一端无论按不按下它都是 1 。这个时候 if(P13==0)的话那么就是 S5 按下了,同理。如果 if(P12==0)的话就是 S6 按下了。
那么判断第三行以及第四行的话都是跟上面所讲述一样,这里不多描述了!
以上是 逐行扫描 的内容!!!但是这个开发板 这样会 出现问题:说明一下这个开发板!不是这个矩阵键盘和知识点的一个问题。这是它内部电路的连接问题 按行扫描的话这个P15口的话可能会一会给高电平或者低电平。(会连接到五线四相步进电机然后BZ连接到蜂鸣器上,因为我们这个蜂鸣器它是无源蜂鸣器,所以当你按行扫描的时候它有可能就会发出声音)
所以建议采用 逐列扫描!
同理!给下面④(P13、P12、P11、P10)个进行赋值。读取上面的④个(P17~P14)
比如说我们要扫描 第一列的话:把 P13 赋值为 (0 低电平) 其它全部给 (1 高电平)那么这一行都可以进行按键扫描了,如果要按 S1 的话 if(P17 == 0) 的话就是 S1 按下了,那么如果 if(P16 == 0) 的话那么就是 S2 按下了,同理~ 。那么第二列也是一样只需要给:P12赋值为低电平,其它给上高点平~~~
单片机IO口的模式
单片机的io口是一种弱上拉的模式~!又被称作是准双向口(input,output) 既可以输入又可以输出,这种就叫做是双向口。但是这种双向口有点问题:这么样才可以达到输入或者是输出呢 ?像我们这种矩阵键盘的话是不是给上,一端是0,然后读取另一头。但是另一头你怎么知道它是一种输入(高电平)呢?它其实也是作为一种输出端(低电平)它既是输出(低电平)也是输入(高电平),那么为什么单片机它的 io 口是默认为高电平呢?是因为它里面拥有一个上拉电阻把低电平变成高电平了 !所以才导致单片机是高电平,还有一个是当口线输出为1的时候驱动能力很弱,允许外部装置将其拉低。当引脚的输出为低电平的时候,它的驱动能力很强,可以吸收相当大的电流。单片机中P1、P2、P3 都是一种弱上拉的一种模式。
准双向口输出如下所示:
提高代码的效率
这样可以固定代码,可以提高自己打代码的一个效率!
代码实现矩阵按键显示对应数字
main.c
#include <REGX52.H> #include "Delay.h" //包含Delay头文件 #include "LCD1602.h" //包含LCD1602头文件 #include "MatrixKey.h" //包含矩阵键盘头文件 unsigned char KeyNum; int main(void) { LCD_Init(); //LCD初始化 LCD_ShowString(1,1,"MatrixKey:"); //LCD显示字符串 while(1) { KeyNum=MatrixKey(); //获取矩阵键盘键码 if(KeyNum) //如果有按键按下 { LCD_ShowNum(2,1,KeyNum,2); //LCD显示键码 } } }
我们需要运用到这个矩阵键盘,所以要在MatrixKey.h当中去进行声明,记得在那个文件加上分号去进行声明,然后在main.c的头文件去进行引用!
再把返回值接到我们所创建的全局变量赋值到KeyNum当中去。
这里if语句当中表达式其实就是如果KeyNum不是为0的话就执行非0即为真执行下面内容。
Delay.c
void Delay(unsigned int xms) { unsigned char i, j; while(xms--) { i = 2; j = 239; do { while (--j); } while (--i); } }
Delay.h
#ifndef __DELAY_H__ #define __DELAY_H__ void Delay(unsigned int xms); #endif
LCD1602.c
1.
#include <REGX52.H> //引脚配置: sbit LCD_RS=P2^6; sbit LCD_RW=P2^5; sbit LCD_EN=P2^7; #define LCD_DataPort P0 //函数定义: /** * @brief LCD1602延时函数,12MHz调用可延时1ms * @param 无 * @retval 无 */ void LCD_Delay() { unsigned char i, j; i = 2; j = 239; do { while (--j); } while (--i); } /** * @brief LCD1602写命令 * @param Command 要写入的命令 * @retval 无 */ void LCD_WriteCommand(unsigned char Command) { LCD_RS=0; LCD_RW=0; LCD_DataPort=Command; LCD_EN=1; LCD_Delay(); LCD_EN=0; LCD_Delay(); } /** * @brief LCD1602写数据 * @param Data 要写入的数据 * @retval 无 */ void LCD_WriteData(unsigned char Data) { LCD_RS=1; LCD_RW=0; LCD_DataPort=Data; LCD_EN=1; LCD_Delay(); LCD_EN=0; LCD_Delay(); } /** * @brief LCD1602设置光标位置 * @param Line 行位置,范围:1~2 * @param Column 列位置,范围:1~16 * @retval 无 */ void LCD_SetCursor(unsigned char Line,unsigned char Column) { if(Line==1) { LCD_WriteCommand(0x80|(Column-1)); } else if(Line==2) { LCD_WriteCommand(0x80|(Column-1+0x40)); } } /** * @brief LCD1602初始化函数 * @param 无 * @retval 无 */ void LCD_Init() { LCD_WriteCommand(0x38);//八位数据接口,两行显示,5*7点阵 LCD_WriteCommand(0x0c);//显示开,光标关,闪烁关 LCD_WriteCommand(0x06);//数据读写操作后,光标自动加一,画面不动 LCD_WriteCommand(0x01);//光标复位,清屏 } /** * @brief 在LCD1602指定位置上显示一个字符 * @param Line 行位置,范围:1~2 * @param Column 列位置,范围:1~16 * @param Char 要显示的字符 * @retval 无 */ void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char) { LCD_SetCursor(Line,Column); LCD_WriteData(Char); } /** * @brief 在LCD1602指定位置开始显示所给字符串 * @param Line 起始行位置,范围:1~2 * @param Column 起始列位置,范围:1~16 * @param String 要显示的字符串 * @retval 无 */ void LCD_ShowString(unsigned char Line,unsigned char Column,char *String) { unsigned char i; LCD_SetCursor(Line,Column); for(i=0;String[i]!='\0';i++) { LCD_WriteData(String[i]); } } /** * @brief 返回值=X的Y次方 */ int LCD_Pow(int X,int Y) { unsigned char i; int Result=1; for(i=0;i<Y;i++) { Result*=X; } return Result; } /** * @brief 在LCD1602指定位置开始显示所给数字 * @param Line 起始行位置,范围:1~2 * @param Column 起始列位置,范围:1~16 * @param Number 要显示的数字,范围:0~65535 * @param Length 要显示数字的长度,范围:1~5 * @retval 无 */ void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length) { unsigned char i; LCD_SetCursor(Line,Column); for(i=Length;i>0;i--) { LCD_WriteData(Number/LCD_Pow(10,i-1)%10+'0'); } } /** * @brief 在LCD1602指定位置开始以有符号十进制显示所给数字 * @param Line 起始行位置,范围:1~2 * @param Column 起始列位置,范围:1~16 * @param Number 要显示的数字,范围:-32768~32767 * @param Length 要显示数字的长度,范围:1~5 * @retval 无 */ void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length) { unsigned char i; unsigned int Number1; LCD_SetCursor(Line,Column); if(Number>=0) { LCD_WriteData('+'); Number1=Number; } else { LCD_WriteData('-'); Number1=-Number; } for(i=Length;i>0;i--) { LCD_WriteData(Number1/LCD_Pow(10,i-1)%10+'0'); } } /** * @brief 在LCD1602指定位置开始以十六进制显示所给数字 * @param Line 起始行位置,范围:1~2 * @param Column 起始列位置,范围:1~16 * @param Number 要显示的数字,范围:0~0xFFFF * @param Length 要显示数字的长度,范围:1~4 * @retval 无 */ void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length) { unsigned char i,SingleNumber; LCD_SetCursor(Line,Column); for(i=Length;i>0;i--) { SingleNumber=Number/LCD_Pow(16,i-1)%16; if(SingleNumber<10) { LCD_WriteData(SingleNumber+'0'); } else { LCD_WriteData(SingleNumber-10+'A'); } } } /** * @brief 在LCD1602指定位置开始以二进制显示所给数字 * @param Line 起始行位置,范围:1~2 * @param Column 起始列位置,范围:1~16 * @param Number 要显示的数字,范围:0~1111 1111 1111 1111 * @param Length 要显示数字的长度,范围:1~16 * @retval 无 */ void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length) { unsigned char i; LCD_SetCursor(Line,Column); for(i=Length;i>0;i--) { LCD_WriteData(Number/LCD_Pow(2,i-1)%2+'0'); } }
LCD1602.h
1.
#ifndef __LCD1602_H__ #define __LCD1602_H__ //用户调用函数: void LCD_Init(); void LCD_ShowChar(unsigned char Line,unsigned char Column,char Char); void LCD_ShowString(unsigned char Line,unsigned char Column,char *String); void LCD_ShowNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length); void LCD_ShowSignedNum(unsigned char Line,unsigned char Column,int Number,unsigned char Length); void LCD_ShowHexNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length); void LCD_ShowBinNum(unsigned char Line,unsigned char Column,unsigned int Number,unsigned char Length);
MatrixKey.c
1.
#include <REGX52.H> #include "Delay.h" /** * @brief 矩阵键盘读取按键键码 * @param 无 * @retval KeyNumber 按下按键的键码值 如果按键按下不放,程序会停留在此函数,松手的一瞬间,返回按键键码,没有按键按下时,返回0 ! */ unsigned char MatrixKey() { unsigned char KeyNumber=0; P1=0xFF;// 1111 1111 全部置高电平默认 P1_3=0; // 矩阵按键第一行扫描 if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=1;} if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=5;} if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=9;} if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=13;} P1=0xFF; P1_2=0; // 矩阵按键第二行扫描 if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=2;} if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=6;} if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=10;} if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=14;} P1=0xFF; P1_1=0; // 矩阵按键第三行扫描 if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=3;} if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=7;} if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=11;} if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=15;} P1=0xFF; P1_0=0; // 矩阵按键第四行扫描 if(P1_7==0){Delay(20);while(P1_7==0);Delay(20);KeyNumber=4;} if(P1_6==0){Delay(20);while(P1_6==0);Delay(20);KeyNumber=8;} if(P1_5==0){Delay(20);while(P1_5==0);Delay(20);KeyNumber=12;} if(P1_4==0){Delay(20);while(P1_4==0);Delay(20);KeyNumber=16;} return KeyNumber; }
注意:在上次的代码当中是没有返回值,但是有参数的。而这次是有返回值且无参数的自定义函数。我们在main函数当中也要定义一个变量来去接收这个返回值💬,我们这里定义的返回值变量是在局部变量所定义的,然后进行return返回到自定义函数当中,要不然它就可能不是初始值了!
P1这些东西都是在 REGX52.H 这个头文件里面,这是在头文件当中定义之中才可以去使用的,不然是不能进行使用的。
MatrixKey.h
#ifndef __MATRIXKEY_H__ #define __MATRIXKEY_H__ unsigned char MatrixKey(); #endif
矩阵按键密码
其它的和上面的一样就是源文件进行改变!
main.c
1.
#include <REGX52.H> #include "Delay.h" #include "LCD1602.h" #include "MatrixKey.h" // 按键作用: S1~S9 设置数字为 1~9, S10定义为数字0, S11用作于是确认按键, S12用作于是取消按键 《《《 S13~S16,我们不去进行使用 unsigned char KeyNum; // 全局变量初始化默认为:0 unsigned int Password,Count; // 如果用6位数字的密码就会超出这个 unsigned int 的一个数值的范围了 0~65535, Count作用:计次,防止输入过多的密码 int main(void) { LCD_Init(); LCD_ShowString(1,1,"Password:"); while(1) { KeyNum=MatrixKey(); if(KeyNum) { if(KeyNum<=10) //如果S1~S10按键按下,输入密码 { if(Count<4) //如果输入次数小于4 { Password*=10; //密码左移一位 : Password = Password * 10 Password+=KeyNum%10; //获取一位密码 : Password = password + KeyNum % 10, 1~9取模10还是为原来的数字~ 获取密码用取模%运算符然后进行赋值 Count++; //计次加一 } LCD_ShowNum(2,1,Password,4); //更新显示 0000 0000 输入第一次(1) 显示0001 》》》 0001 0010 输入第二次(2) 显示0012 } if(KeyNum==11) //如果S11按键按下,确认 ----注意:这里不进行消抖的原因是:模块化编程的时候已经进行消抖了 { if(Password==2345) //如果密码等于正确密码 --------------------------- 定义密码 { LCD_ShowString(1,14,"OK "); // 显示OK Password=0; // 密码清零 Count=0; // 计次清零 LCD_ShowNum(2,1,Password,4); // 更新显示 } else //否则 { LCD_ShowString(1,14,"ERR"); //显示ERR Password=0; // 密码清零 Count=0; // 计次清零 LCD_ShowNum(2,1,Password,4); //更新显示 } } if(KeyNum==12) //如果S12按键按下,取消 { Password=0; // 密码清零 Count=0; // 计次清零 LCD_ShowNum(2,1,Password,4); //更新显示 } } } }