STM32矩形(矩阵)按键(键盘)输入控制LED灯 ——4*4矩阵按键源码解析

news2024/11/16 9:36:59

本文基于标准函数库的工程实现stm32F103C8T6使用4*4的矩阵按键控制LED灯的亮灭及闪烁等功能。

程序源码:链接:https://pan.baidu.com/s/1_MPhvMduKCTP0MPG-Gtw3A?pwd=2syk 
提取码:2syk

文章目录

一、矩形键盘介绍

1、硬件电路基本原理

2、两种识别方法介绍

3、硬件接线即使用

二、程序源码

1、矩阵键盘源码说明

2、主函数源码

三、实验现象


一、矩形按键介绍

1、硬件电路基本原理

矩阵键盘意思是指按键的电路排列类似于矩阵的按键,而不是按键的排列外表呈矩阵状。矩阵式键盘用N条I/O线作为行线,N条I/O线作为列线,构成了一个具有N*N个按键的矩阵按键。由图中可见,这个4*4一共16个按键的矩阵按键只需要接8个IO口,若采用单个按键接法,却需要16个IO口,由此可见采用矩阵键盘可以大大提高IO口的利用率。

其原理图如下:

2、两种识别方法介绍

一般对于矩阵键盘,有扫描法和反转法识别按键状态。

2.1  反转法

反转法找出每个按键的码,程序中判断8个引脚的状态,进行对比,判断出是那哪按键被按下,每个按键的码值由行码和列码构成。

我们以行为输入,列为输出。这里行输入端需要使用上拉输入(即四个行引脚均为高电平),输出需要使用推挽输出(输出为低电平),然后读取四个行输入的状态,若读到某一行为低电平时,则对应行被按下,这样就可以获取到行码。

用同样的方法,将行跟列的输出输入对调再进行判断,这样就可以得到行码,结合行列码,我们就可以知道是哪个按钮被按下。

2.2  扫描法

扫描法:令所有的引脚都为高电平(行为输出,使用推挽输出高电平,列为输入,使用上拉电阻输入高电平),这时我们配置第一行为0,然后判断各列的输入状态,如某一列出出现低电平,则可判断出是第一行的某一列被按下;如果没有出现低电平,则令第二行引脚为0其余引脚为高电平,判断各列的输入状态;如此反复,若没有输入没有识别到低电平就继续对三四行进行判断,如此就可以对四行四列进行判断。

3、硬件接线及使用

由上图可知整个4*4的矩阵键盘一共为8个引脚,其中C1、C2、C3、C4为列,R1、R2、R3、R4为行,这里将行接到PA0,PA1,PA2,PA3 ,列接到PA4,PA5,PA6,PA7。

二、程序源码

1、矩阵键盘源码说明

这里我们用扫描法进行矩阵按钮的识别,详细如下:

在函数初始化的时候给矩阵按钮的所有引脚配置成1(高电平),然后在实行逐行扫描,先令第一行为0(低电平),这是第一行的四个按钮进行判断(读取GPIO的引脚电平),若有按钮为0(低电平)时,即按钮被按下。当按钮被按下时,会将对应的键码传送到变量key中。

由于是机械按键,在按钮被按下的时候要进行消抖,然后在按钮按下的状态进行一个循环读取,当按钮松开后再对后面的按钮进行扫描,防止多个按钮同时按下时出现误判的情况。

第一行扫描完成且没有按钮被按下后,按照上面的方法对剩余的三行进行扫描。

将识别的子程序放在while循环函数中,可进行无限循环判断,这样我们对矩阵按钮的识别函数就完成了。详细程序看下方代码即后面的注释

#include "stm32f10x.h"
#include "key.h"
#include "Delay.h"

u8 FLAG = 0;  //FLAG这个变量为1时,证明有按钮正在被按下

void KEY_4x4_Init(void)     //键盘IO口配置及初始化
{
 	GPIO_InitTypeDef GPIO_InitStructure;       	
 	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA,ENABLE);      
	//使用GPIOA的0,1,2,3引脚为行 R1~R4对应矩形按钮的5,6,7,8引脚
	GPIO_InitStructure.GPIO_Pin  = KEY_HANG;  //行  0123
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;     //使用推挽输出
    GPIO_Init(GPIOA, &GPIO_InitStructure);
	GPIO_SetBits(GPIOA,KEY_HANG);   //令GPIO的0,1,2,3引脚输出为1
	/********************************************************************/
	GPIO_InitStructure.GPIO_Pin  = lie1|lie2|lie3|lie4;         
	//使用GPIOA的4,5,6,7引脚为列  C1~C4对应矩形按钮的4,3,2,1引脚
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;        //使用上拉输入
	GPIO_Init(GPIOA, &GPIO_InitStructure);
	GPIO_SetBits(GPIOA,lie1|lie2|lie3|lie4);	//令GPIO的4,5,6,7引脚输出为1	
}
//端口配置后,GPIOA的所有端口输出均为高电平
void KEY_Scan(u8 *key) 
{	 	
	GPIO_Write(GPIOA,0x00fe); //第一行  0000 0000 1111 1110   低四位为行   高四位为列
	/*令第一行输出为低电平,这时再判断4个列的输入状态的值,
	在按键没有被按下时,四列输出都是1,如果其中一列变为1,则可以判断S1按钮被按下*/
	if((lie1_Input==0)||(lie2_Input==0)||(lie3_Input==0)||(lie4_Input==0))   
	{
		Delay_ms(10);  //去抖动 
		
		if(lie1_Input==0)   //如果第一列被按下
		{
			FLAG = 1;    //按钮判断变量
			*key = 1;   //输出key的值为1传递到主函数,当值为x时,则说明按钮Sx被按下
				/*在变量前增加*,为c语言中的指针操作,使用指针进行地址传递,在子函数中修改的值在主函数中可以利用地址读取,
				而不需要利用子函数返回值,然后主函数在增加一个变量进行接收*/
			while(!GPIO_ReadInputDataBit(GPIOA,lie1));   
				//当按钮处于被按下的状态的时候,程序一直卡在循环读取按钮的状态,避免多按钮同时按下时读取错误
		}
		else if(lie2_Input==0)
		{
			FLAG = 1;
			*key = 2;
			while(!GPIO_ReadInputDataBit(GPIOA,lie2));
		}
		else if(lie3_Input==0)
		{
			FLAG = 1;
			*key = 3;
			while(!GPIO_ReadInputDataBit(GPIOA,lie3));
		}
		else if(lie4_Input==0)
		{
	    	FLAG = 1;
			*key = 4;
			while(!GPIO_ReadInputDataBit(GPIOA,lie4));
		}
		else  //如果第一行四列中没有按钮被按下
		{
			FLAG = 0;
			GPIO_Write(GPIOA,0x00ff);
		}
	}//第一行判断完成,这是我们判断第二行
	GPIO_Write(GPIOA,0x00fd);    //第二行         
	if((lie1_Input==0)||(lie2_Input==0)||(lie3_Input==0)||(lie4_Input==0))
	{
		Delay_ms(10);//去抖动 
		if(lie1_Input==0)
		{
			FLAG = 1;
			*key = 5;
			while(!GPIO_ReadInputDataBit(GPIOA,lie1));
		}
		else if(lie2_Input==0)
		{
	     	FLAG = 1;
			*key = 6;
			while(!GPIO_ReadInputDataBit(GPIOA,lie2));
		}
		else if(lie3_Input==0)
		{
	    	FLAG = 1;
			*key = 7;
			while(!GPIO_ReadInputDataBit(GPIOA,lie3));
		}
		else if(lie4_Input==0)
		{	 
	    	FLAG = 1;
			*key = 8;
			while(!GPIO_ReadInputDataBit(GPIOA,lie4));
		}
		else 
		{
			FLAG = 0;
			GPIO_Write(GPIOA,0x00ff);
		}
	}
	GPIO_Write(GPIOA,0x00fb);//第三行
	if((lie1_Input==0)||(lie2_Input==0)||(lie3_Input==0)||(lie4_Input==0))
	{
		Delay_ms(10);//去抖动 
		if(lie1_Input==0)
		{
			FLAG = 1;
			*key = 9;
			while(!GPIO_ReadInputDataBit(GPIOA,lie1));
		}
		else if(lie2_Input==0)
		{
	     	FLAG = 1;
			*key = 10;
			while(!GPIO_ReadInputDataBit(GPIOA,lie2));
		}
		else if(lie3_Input==0)
		{
	    	FLAG = 1;
			*key = 11;
			while(!GPIO_ReadInputDataBit(GPIOA,lie3));
		}
		else if(lie4_Input==0)
		{
	    	FLAG = 1;
			*key = 12;
			while(!GPIO_ReadInputDataBit(GPIOA,lie4));
		}
		else 
		{
			FLAG = 0;
			GPIO_Write(GPIOA,0x00ff);
		}
	}
	GPIO_Write(GPIOA,0x00f7);//第四行
	if((lie1_Input==0)||(lie2_Input==0)||(lie3_Input==0)||(lie4_Input==0))
	{
		Delay_ms(10);//去抖动 
		if(lie1_Input==0)
		{
			FLAG = 1;
			*key = 13;
			while(!GPIO_ReadInputDataBit(GPIOA,lie1));
		}
		else if(lie2_Input==0)
		{
	     	FLAG = 1;
			*key = 14;
			while(!GPIO_ReadInputDataBit(GPIOA,lie2));
		}
		else if(lie3_Input==0)
		{
	    	FLAG = 1;
			*key = 15;
			while(!GPIO_ReadInputDataBit(GPIOA,lie3));
		}
		else if(lie4_Input==0)
		{
	    	FLAG = 1;
			*key = 16;
			while(!GPIO_ReadInputDataBit(GPIOA,lie4));
		}
		else 
		{
			FLAG = 0;
			GPIO_Write(GPIOA,0x00ff);
		}
	}
	
}





2、主函数源码

这里用上GPIOB的10,11,12三个引脚接LED彩灯模块,对应为R,G,B(红,绿,蓝)三色,主函数中有两个子函数段,一个为端口初始化,另一个则是LED闪烁变换的程序段,其中函数的参数为闪烁间隔的时间。其现象为:按下S1,红灯亮;按下S2,绿灯亮;按下S3,蓝灯亮;按下S4,熄灭所有灯;按下S5,以0.3s的间隔RGB轮流闪烁一次;按下S6,以0.5s的间隔RGB循坏闪烁3次

#include "stm32f10x.h"                  // Device header
#include "Delay.H"
#include "key.H"

/****************************************************/
/*引脚使用说明:									*/
/*				矩形键盘: C1~C4对应GPIOA 4,5,6,7   */  
/*						   R1~R4对应GPIOA 0,1,2,3   */
/*				LED中的R,G,B对应GPIOB 10,11,12      */
/****************************************************/


void GPIO_Config(void)    //对GPIOB进行配置初始化
{
	RCC_APB2PeriphClockCmd (RCC_APB2Periph_GPIOB,ENABLE );
	GPIO_InitTypeDef GPIO_Initstructure;
	GPIO_Initstructure .GPIO_Mode =GPIO_Mode_Out_PP;
	GPIO_Initstructure .GPIO_Pin=0X1D00 ;    //0001 1100 0000 0000
	GPIO_Initstructure .GPIO_Speed =GPIO_Speed_50MHz ;
	GPIO_Init (GPIOB ,&GPIO_Initstructure );
}

void LED_RGB(float  time)
{	
	u16 t =time *1000;		
	GPIO_Write (GPIOB ,0X0400);  // 0000 0100 0000 0000
	Delay_ms (t);
	GPIO_Write (GPIOB ,0X0800);  // 0000 1000 0000 0000
	Delay_ms (t);
	GPIO_Write (GPIOB ,0X1000);  // 0001 0000 0000 0000
	Delay_ms (t);
	GPIO_Write (GPIOB ,0X0000);
}

u8 key=0;
u8 k =0;
int main(void)
{
	GPIO_Config ();
	KEY_4x4_Init();
	while(1)
	{
		KEY_Scan (&key);
		if(FLAG == 1)    //按键按下
		{
			FLAG = 0;
			if(key==1)  //按下S1,亮红灯	
			{
				GPIO_SetBits (GPIOB ,GPIO_Pin_10 );
			}	
			else if(key==2)  //按下S2,亮绿灯
			{
				GPIO_SetBits(GPIOB,GPIO_Pin_11 );
			}
			else if(key==3)		//按下S3,亮蓝灯
			{
				GPIO_SetBits(GPIOB,GPIO_Pin_12 );
			}
			else if(key==4)   //按下S4,关闭所有灯
			{
				GPIO_Write (GPIOB,0X0000);   //0000 0000 0000 0000
			}
			else if(key==5)
			{
				LED_RGB (0.3);   //子函数参数为时间,单位为s
			}
			else if(key==6)
			{
				for(k=0;k<3;k++)
				{
				LED_RGB (0.5);
				}
			}		
		}
	}
}

三、实验现象展示

矩阵按键

本人也是初学STM32,在学习的同时以初学者的角度来理解说明,如果文章有错误的地方,请大家多多指正,谢谢!!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1420156.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Sketch 英文转中文:轻松搞定

Sketch版本的转换一直是每个人的关键问题。现在UI设计领域有很多UI设计软件&#xff0c;但大部分都是英文版。对于国内英语基础差的设计师来说&#xff0c;使用这样的软件无形中增加了工作量&#xff0c;往往需要在设计编辑的同时查阅翻译。中文Sketch版本替代即时设计详细介绍…

动手学RAG:汽车知识问答

原文&#xff1a;动手学RAG&#xff1a;汽车知识问答 - 知乎 Part1 内容介绍 在自然语言处理领域&#xff0c;大型语言模型&#xff08;LLM&#xff09;如GPT-3、BERT等已经取得了显著的进展&#xff0c;它们能够生成连贯、自然的文本&#xff0c;回答问题&#xff0c;并执行…

常用网址备份

阿里git下载镜像 (npmmirror.com 主页有其他资源)https://registry.npmmirror.com/binary.html?pathgit-for-windows/

【Java】Spring的APO及事务

今日目标 能够理解AOP的作用 能够完成AOP的入门案例 能够理解AOP的工作流程 能够说出AOP的五种通知类型 能够完成"测量业务层接口万次执行效率"案例 能够掌握Spring事务配置 一、AOP 1 AOP简介 问题导入 问题1&#xff1a;AOP的作用是什么&#xff1f; 问题2&am…

Phoncent博客,探索Rie Kudan的GPT创作之举

近日&#xff0c;大家都在谈论日本作家Rie Kudan&#xff0c;她凭借其小说《东京共鸣塔》&#xff08;"Tokyo-to Dojo-to"&#xff09;荣获了日本极具声望的芥川奖。这本小说引起了广泛的讨论和思考&#xff0c;因为令人惊讶的是&#xff0c;Kudan在其中直接引用了人…

Python(19)Excel表格操作Ⅰ

目录 导包 读取EXCEL文件 1、获取worksheet名称 2、设定当前工作表 3、输出目标单元格数据 4、工作表.rows&#xff08;行&#xff09; 5、工作表.columns&#xff08;列&#xff09; 小结 导包 要想使用 python 操作 Excel 文件&#xff0c;应当导入 openpyxl 包。在…

Java 面试题之 IO(二)

字符流 文章目录 字符流Reader&#xff08;字符输入流&#xff09;Writer&#xff08;字符输出流&#xff09; 文章来自Java Guide 用于学习如有侵权&#xff0c;立即删除 不管是文件读写还是网络发送接收&#xff0c;信息的最小存储单元都是字节。 那为什么 I/O 流操作要分为字…

Jupyter notebook文件默认存储路径以及更改方法

目录 1、文件默认存储路径怎么查&#xff1f;2、文件默认存储路径怎么改&#xff1f; 转自&#xff1a;https://blog.csdn.net/fengyeer20120/article/details/109483362 初次使用Jupyter Notebook&#xff0c;确实好用啊&#xff01;但安装Anaconda后&#xff0c;打开Jupyter …

WebSocket 整合 记录用法

WebSocket 介绍 WebSocket 是基于tcp的一种新的网络协议,可以让浏览器 和 服务器进行通信,然后区别于http需要三次握手,websocket只用一次握手,就可以创建持久性的连接,并进行双向数据传输 Http和WebSocket的区别 Http是短连接,WebSocket’是长连接Http通信是单向的,基于请求…

个人建站前端篇(一)项目准备初始化以及远程仓库连接

云风的知识库 云风网前端重构&#xff0c;采用vue3.0vite antd框架&#xff0c;实现前后端分离&#xff0c;实现网站的SEO优化&#xff0c;实现网站的性能优化 vite创建vue项目以及前期准备 Vite 需要 Node.js 版本 18&#xff0c;20。然而&#xff0c;有些模板需要依赖更高…

su模型库免费下载哪家好?

选择SU模型库免费下载的网站&#xff0c;需要根据个人的需求和偏好进行评估。以下是一些热门的SU模型库免费下载网站&#xff0c;供您参考&#xff1a; ①建e网&#xff1a;这是一个专业的室内设计资源平台&#xff0c;包括各种类型的SU模型&#xff0c;如家装、公装、商业空间…

linux监控工具

官方吹的牛逼 LATEST BLOG POST: On the same workload, Netdata uses 35% less CPU, 49% less RAM, 12% less bandwidth, 98% less disk I/O, and is 75% more disk space efficient on high resolution metrics storage, while providing more than a year of overall reten…

vue3-深入组件-透传属性

透传属性 &#xff08;透传 attribute&#xff09; 什么是透传属性&#xff08;透传 attribute&#xff09;? 传递给一个组件&#xff0c;却没有被该组件声明为 props 或 emits 的 attribute 或者是事件监听器&#xff0c;例如 class style id 等。 属性继承 当一个组件以单…

面试:问js的forEach和map的区别

前言 为什么要写这么一篇文章&#xff0c;原因是今天下午水群的时候&#xff0c;有个小伙伴分享自己的面试题&#xff0c;其中一个是foreach和map的区别&#xff0c;其实是蛮简单&#xff0c;蛮基础的一道题&#xff0c;但是他是这么回答的 一个会改变原数组&#xff0c;一个不…

GitCode|部分项目开源代码

1.EasyKeyboard 基于MFC的简单软键盘&#xff0c;使用vs2017开发 PangCoder / EasyKeyboard GitCode基于Windows平台的软键盘&#xff0c;使用VS2017开发&#xff0c;使用MFC框架https://gitcode.net/qq_36251561/easykeyboard 2.EncoderSimulator 基于WPF应用的编码器模拟工…

什么是客户关系管理?流程是什么样的?

客户关系管理的目的是什么? 客户关系管理(CRM)使企业能够降低成本并增加利润。CRM系统用于组织、自动化和跟踪业务流程&#xff0c;例如潜在客户生成、营销、预测、销售、投资回报率测量和客户服务。因此&#xff0c;CRM系统的目的是促进这些领域的整合&#xff0c;并衡量和跟…

Xds (eXtensible Discovery Service)理解

Xds &#xff08;eXtensible Discovery Service&#xff09;理解 概念介绍 XDS&#xff1a;全称是 eXtensible Discovery Service&#xff0c;中文译为”可扩展的服务发现“。是一个扩展性的协议&#xff0c;可以用于配置和控制 Envoy 代理的各个方面&#xff0c;包括路由规则…

异步任务的一些思考

前言 XXL-Job部署教程 项目中&#xff0c;必然少不了数据的导入导出&#xff0c;针对数据的导入导出简单复盘一下。 为了不占用资源消耗时间&#xff0c;影响用户体验&#xff0c;大量数据的导入导出一般都是异步执行 导入的时候&#xff0c;如果数据量很大&#xff0c;一次…

研发日记,Matlab/Simulink避坑指南(五)——CAN解包 DLC Bug

文章目录 前言 背景介绍 问题描述 分析排查 解决方案 总结 前言 见《研发日记&#xff0c;Matlab/Simulink避坑指南&#xff08;一&#xff09;——Data Store Memory模块执行时序Bug》 见《研发日记&#xff0c;Matlab/Simulink避坑指南(二)——非对称数据溢出Bug》 见《…