51单片机学习笔记-4矩阵键盘

这篇具有很好参考价值的文章主要介绍了51单片机学习笔记-4矩阵键盘。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

4 矩阵键盘

[toc]

注:笔记主要参考B站江科大自化协教学视频“51单片机入门教程-2020版 程序全程纯手打 从零开始入门”。
注:工程及代码文件放在了本人的Github仓库。


4.1 矩阵键盘介绍

在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式。采用逐行或逐列的“扫描”,就可以读出任何位置按键的状态。

  • 数码管扫描(输出扫描)
    原理:显示第1位→显示第2位→显示第3位→……,然后快速循环这个过程,最终实现所有数码管同时显示的效果。
  • 矩阵键盘扫描(输入扫描)
    原理:读取第1行(列)→读取第2行(列) →读取第3行(列) → ……,然后快速循环这个过程,最终实现所有按键同时检测的效果。但同时按下多个按键可能会造成误判。
  • 以上两种扫描方式的共性:节省I/O口
51单片机学习笔记-4矩阵键盘
图4-1 矩阵键盘实物图及原理图

比如,从上图所示,上四个接口表示行、下四个接口表示列。下面介绍具体的扫描过程:

  • 逐行扫描:
    先将某一行置0,此时若该行有按键按下,那该行上对应的列也就为0;以此重复所有的行,即完成一次扫描。
  • 逐列扫描:
    将“逐行扫描”的行、列互换即可。

注:由于开发板IO资源有限,有一些开发板会把蜂鸣器IO口与矩阵键盘IO口复用,此时看情况更换扫描方式。

单片机的IO口是一种弱上拉(具有较弱的拉高电平或电位的能力)。关于其是如何实现双向口配置的,可以参考STC官方的器件手册“I/O口各种不同的工作模式及其配置介绍”。

4.2 实践:读取矩阵键盘

需求:在LCD1602第一行显示字符串“Matrix key:”,第二行按照按键丝印编号显示按下了哪个键(松开触发)。

为了代码简洁,编写了单独的矩阵按键检测子函数“MatrixKey.h”,并将其单独作为一个文件独立出来。当然,函数声明和定义应该分别放在头文件、源文件中,但由于函数太少,所以为了方便管理我将函数较少的源文件合并在头文件中。下面是工程的组织架构:

51单片机学习笔记-4矩阵键盘
图4-2 “读取矩阵键盘”实验代码调用关系

代码展示:
- main.c

#include <REGX52.H>
#include "LCD1602.h"
#include "Delay.h"
#include "MatrixKey.h"

void main(){
  int key = 0;
  // 初始化LCD显示
  LCD_Init();
  LCD_ShowString(1,1,"Matrix key:");
  LCD_ShowString(2,1,"S00");
  while(1){
    key = MatrixKey();
    if(key){LCD_ShowNum(2,2,key,2);};
  }
}

- MatrixKey.h

#ifndef __MATRIXKEY_H__
#define __MATRIXKEY_H__

#include <REGX52.H>
#include "Delay.h"
/**
  * @brief :按行扫描一次,检测矩阵键盘被按下的按键(松开返回)
  * @param :无
  * @retval :按下按键的标号(与丝印相同),若没有按下就返回0。
  * 注:若按键按下不松开,会一直停留在此函数,直到按键松开后。
 */
unsigned char MatrixKey(){
  unsigned char key = 0;

  // 扫描第一行
  P1 = 0xff;  P1_7 = 0;
  if(!P1_3 && P1_2 && P1_1 && P1_0){
    Delay(10);    // 按下消抖
    while(!P1_3); // 等着松开
    Delay(10);    // 松开消抖
    return key = 1;
  }else if(P1_3 && !P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_2); Delay(10);
    return key = 2;
  }else if(P1_3 && P1_2 && !P1_1 && P1_0){
    Delay(10); while(!P1_1); Delay(10);
    return key = 3;
  }else if(P1_3 && P1_2 && P1_1 && !P1_0){
    Delay(10); while(!P1_0); Delay(10);
    return key = 4;
  }
  // 扫描第二行
  P1 = 0xff;  P1_6 = 0;
  if(!P1_3 && P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_3); Delay(10);
    return key = 5;
  }else if(P1_3 && !P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_2); Delay(10);
    return key = 6;
  }else if(P1_3 && P1_2 && !P1_1 && P1_0){
    Delay(10); while(!P1_1); Delay(10);
    return key = 7;
  }else if(P1_3 && P1_2 && P1_1 && !P1_0){
    Delay(10); while(!P1_0); Delay(10);
    return key = 8;
  }
  // 扫描第三行
  P1 = 0xff;  P1_5 = 0;
  if(!P1_3 && P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_3); Delay(10);
    return key = 9;
  }else if(P1_3 && !P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_2); Delay(10);
    return key = 10;
  }else if(P1_3 && P1_2 && !P1_1 && P1_0){
    Delay(10); while(!P1_1); Delay(10);
    return key = 11;
  }else if(P1_3 && P1_2 && P1_1 && !P1_0){
    Delay(10); while(!P1_0); Delay(10);
    return key = 12;
  }
  // 扫描第四行
  P1 = 0xff;  P1_4 = 0;
  if(!P1_3 && P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_3); Delay(10);
    return key = 13;
  }else if(P1_3 && !P1_2 && P1_1 && P1_0){
    Delay(10); while(!P1_2); Delay(10);
    return key = 14;
  }else if(P1_3 && P1_2 && !P1_1 && P1_0){
    Delay(10); while(!P1_1); Delay(10);
    return key = 15;
  }else if(P1_3 && P1_2 && P1_1 && !P1_0){
    Delay(10); while(!P1_0); Delay(10);
    return key = 16;
  }
  
  return key;
}

#endif

- Dealy.h

#ifndef __DEALY_H_
#define __DEALY_H_

// 延时cycles ms,晶振@11.0592MHz
void Delay(unsigned int cycles){
  unsigned char i, j;
  do{
    i = 2;
    j = 199;
    do{
      while (--j);
    }while (--i);
  }while(--cycles);
}

#endif

- 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);

#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');
	}
}

4.3 实践:矩阵键盘密码锁

需求:
实现一个矩阵键盘密码锁。S1~S9代表数字1~9,S10代表数字0,S11代表“确认”,S12代表“取消”。

  • LCD第一行左侧显示“Password:”,第二行显示输入的四位密码,每次按下按键会依次左移数字。
  • 按下“确认”时,判断密码是否正确,并显示在LCD第一行右侧“Err!”、“True!”,同时密码归位。
  • 按下“取消”时,当前密码归零。
51单片机学习笔记-4矩阵键盘
图4-3 “矩阵键盘密码锁”实验代码调用关系

- main.c

#include <REGX52.H>
#include "MatrixKey.h"
#include "LCD1602.h"

void main(){
  unsigned int correct = 123; // 正确的密码
  unsigned int password = 0; // 当前的密码值
  unsigned char key = 0; // 矩阵键盘当前按下的值
  // 显示屏初始化
  LCD_Init();
  LCD_ShowString(1,1,"Password:");
  LCD_ShowNum(2,1,0,4);
  while(1){
    key = MatrixKey();
    if(key==11){ // 按下确认键,则进行判断
      if(password == correct)
        LCD_ShowString(1,12,"True!");
      else
        LCD_ShowString(1,13,"Err!");
    }else if(key==12){ // 按下取消键,密码归零
      LCD_ShowString(1,12,"     "); // 清除对错提示
      password = 0;
      LCD_ShowNum(2,1,password,4);
    }else if(key>=1 && key<=10){ // 其他正常的数字键
      LCD_ShowString(1,12,"     "); // 清除对错提示
      key = (key==10) ? 0 : key; // 或者也可以对10取余
      password = (password%1000) * 10 + key;
      LCD_ShowNum(2,1,password,4);
    }
  }
}

其余的 MatrixKey.hDelay.hLCD1602.hLCD1602.c 模块均与4.2节实验相同。文章来源地址https://www.toymoban.com/news/detail-464532.html

到了这里,关于51单片机学习笔记-4矩阵键盘的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 51单片机入门——矩阵键盘(附51代码)

    硬件如图非常简单,将一个4*4的矩阵键盘的8个管脚引到端子上,在连接到8个I/O口上,ARRAY_H代表着行,ARRAY_L代表着列,当行与列的电平都置低的时候,就选中的相应的矩阵按键,比如当s1按下时,ARRAY_H1会置低,其他ARRAY_H给高电平,那么选中的就是第一行,然后到列,ARRAY_

    2024年02月11日
    浏览(43)
  • 51单片机(六)矩阵键盘和矩阵键盘密码锁

    ❤️ 专栏简介:本专栏记录了从零学习单片机的过程,其中包括51单片机和STM32单片机两部分;建议先学习51单片机,其是STM32等高级单片机的基础;这样再学习STM32时才能融会贯通。 ☀️ 专栏适用人群 :适用于想要从零基础开始学习入门单片机,且有一定C语言基础的的童鞋

    2024年02月04日
    浏览(53)
  • 6.51单片机之矩阵键盘

    👻 1.矩阵键盘的介绍 在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式。 采用逐行或逐列的“扫描”,就可以读出任何位置按键的状态。 结构:在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式。在矩阵式键盘中,每条

    2024年02月07日
    浏览(44)
  • 51单片机矩阵键盘——数码管显示

    当我们熟悉了数码管的位选与段选,并了解的矩阵键盘的扫描之后就可以编写程序了。 按下矩阵键盘S1并松开,数码管第一位(LED8)显示0;按下矩阵键盘S2并松开,数码管第一位显示1;...按下矩阵键盘S16并松开,数码管第一位显示F; 矩阵键盘扫描(输入扫描)     原理:

    2024年02月11日
    浏览(55)
  • 51单片机实现矩阵键盘密码锁

    使用51单片机的矩阵键盘模块以及led1602显示屏,实现模拟密码锁。 当程序烧录到单片机中后,led1602屏幕会显示文字。 第一行会显示单词“PASSWORD”,第二行显示4个0,表示我们要写入的四位密码,每位默认为0。 矩阵键盘前两行与第三行的前两个分别代表输入1-9与0,第三行第

    2024年02月03日
    浏览(57)
  • 51单片机矩阵键盘——LCD1602显示

    本次的实验需要用到LCD1602液晶屏,需要用到LCD1602.c 与LCD1602.h文件链接: https://download.csdn.net/download/YLG_lin/86404949 使用矩阵键盘随意按下按键,LCD1602第二行前两位就会显示对应按键按下的数字。 在键盘中按键数量较多时,为了减少I/O口的占用,通常将按键排列成矩阵形式 采用

    2023年04月09日
    浏览(44)
  • 如何在51单片机上实现矩阵键盘功能

    本文详细介绍了在51单片机上实现矩阵键盘功能的方法,包括按键检测、消抖处理和键值映射等关键步骤。通过本文的学习,你将能够掌握在嵌入式系统中处理矩阵键盘输入的技术。

    2024年04月10日
    浏览(98)
  • 51单片机矩阵键盘扫描及使用方法

             矩阵键盘 ,也称矩阵按键,是为了节约单片机IO口占用所引入的一种外设。 (图片截取至普中A2开发板原理图) (图片截取至普中A2开发板实物图)         我们知道,一个独立按键需要1个IO口。但是如果我们需要大量的按键,则需要大量的IO口,但是单片机

    2024年02月02日
    浏览(37)
  • 51单片机设计16个按键的矩阵键盘

    首先如图所示导入各个元器件,并连线。 因为P3口连接着矩阵键盘,所以上来要先付个初值,从高位到低位H0,H1,H2,H3,L0,L1,L2,L3的值是11110000,高位全是1,低位全是0,十六进制就是0xF0,也就是P3=0xF0。如果有按键按下了,肯定就不是这个值了。 比如说: 第一行789+中的任何一个按

    2024年02月05日
    浏览(45)
  • 基于51单片机的矩阵键盘(线反转法)

    矩阵键盘使用的方法与独立按键类似,但是比独立按键节省I/O,同时使用方法变得相对复杂了,首先展示硬件连接。 上图所示就是4X4矩阵键盘的常见接法,按行来看,每个按键的左端接在一起,按列来看,每个按键的右端接在一起,共占用8个I/O,接到51单片机的P1端口。 对于

    2024年02月11日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包