写点什么

【51 单片机】矩阵键盘

作者:泽En
  • 2022 年 5 月 02 日
  • 本文字数:6948 字

    阅读完需:约 23 分钟

【51单片机】矩阵键盘

🚀write in front🚀

🔎​​Hello,大家好我是泽 En,一起共同学习,多多指教(●'◡'●)🔎

🎁欢迎各位→点赞👍 + 收藏⭐️ + 留言📝

✉️我们并非登上我们所选择的舞台,演出并非我们所选择的剧本📩

💬总结:希望你看完之后,能对你有所帮助,不足请指正!共同学习交流🖊

矩阵键盘介绍

在键盘中按键数量较多时,为了减少 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

#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

#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

#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


#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;        // 全局变量初始化默认为:0unsigned 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); //更新显示 } } }}
复制代码


发布于: 刚刚阅读数: 2
用户头像

泽En

关注

好像没有😅 2022.01.29 加入

CSDN嵌入式领域新星创作者、2021年度博客之星物联网与嵌入式开发TOP5、2022博客之星TOP100 掘金创作者

评论

发布
暂无评论
【51单片机】矩阵键盘_5月月更_泽En_InfoQ写作社区