我在高职教STM32——GPIO入门之按键输入(2)

news2025/1/15 6:47:30

        大家好,我是老耿,高职青椒一枚,一直从事单片机、嵌入式、物联网等课程的教学。对于高职的学生层次,同行应该都懂的,老师在课堂上教学几乎是没什么成就感的。正因如此,才有了借助 CSDN 平台寻求认同感和成就感的想法。在这里,我准备陆续把自己花了很多心思的教学设计分享出来,主要面向广大师生朋友,单片机老鸟就略过吧。欢迎点赞+关注,各位的支持是本人持续输出的动力,多谢多谢!

        前面,我们介绍了STM32的IO口作为输出的使用,这一章,我们将向大家介绍如何使用IO口作为输入。在本章中,我们将利用开发板上的按键来控制LED的亮灭。通过本章的学习,我们将明白按键的电路原理,了解按键消抖是怎么回事,巩固GPIO的初始化配置,学习GPIO端口输入函数等知识。

【学习目标】

  1. 了解按键防抖、锁存的方法
  2. 巩固GPIO初始化的过程,独立完成代码编写
  3. 理解按键单击、双击、长按的程序算法

        按键是初学嵌入式的第一类输入器件,入门不难,但是一旦按法多样化(单击/双击/长按),或是结合其他被控器件,就需要用上中断、定时器、状态机等知识,难度也就上来了。本章还是基于GPIO输入电平的传统方法来按键,计划分两个部分,本文是第二部分。

三、按键的单击/双击/长按

        电子产品上的轻触按键,除了单击,双击和长按也是比较普遍的输入方式,这样可以在同一个按键上实现更多的控制效果,鼠标就是典型的例子。上一节的程序仅能检测单击这个动作,这一节,我们就来学习一下如何通过一定算法,把这三种动作都能检测出来。

3.1 编程要点

        我们来分析三种动作的按键IO口电平变化,如图7所示,看看各自有何特征。首先,无论何种动作,都是从按下这个动作开始的。接下来,就是要看按下的时长了,如果超过了长按标准(图中的S3,如2s),那么毫无疑问肯定是长按了,也就排除了单击和双击。因此,从逻辑上来看,判断出长按所需的条件是最少的。

        其次,如果按下的时长小于S3,也就是说没到长按标准就松开了,那么就只能是单击或双击其中一种可能。这时,看的就是松开的时长了,如果超过了两次单击间隔标准(图中的D1,如250ms),那么就不是双击了,只能是单击。反之,就是双击。所以,单击或双击的判断依据就变成了“按下时长小于S3 且 松开时长是否小于D1”这样的双重条件了。

        最后,还有一种可能,那就是无动作。当然,发生这种情况的条件是没有按下这个动作发生。这种情况其实是程序的初态,或者说是完成一次按键动作后应该回归的状态。至此,本程序的目的就是要通过一系列条件判断得到“单击/双击/长按/无动作”其中一种结果。

图7 单击/双击/长按对应的IO口电平变化

3.2 代码剖析

        本实验的硬件电路和工程文件清单跟上一个实验一样,但文件中的代码发生了较大变化,因此我们将上一个实验的工程文件另存一份再进行编写,这样不会混淆。需要说明的是,这里为了节省篇幅和排版需要,一些与之前重复的代码做了省略,阅读时请注意,完整的源码请阅读本实验配套的工程。

1. key.h文件源码

        如代码清单4所示,这个头文件了增加了很多跟按键动作有关的宏,其实就是把每个按键的每一种可能都起好名字,编好数字。当然,还有一个用来扫描按键动作的函数声明 Key_Scan(),其代码也是本实验的重点。

//-------------------------------------------------------
// 代码清单4:补充与完善后的key.h
//-------------------------------------------------------
 
#ifndef  _KEY_H_
#define  _KEY_H_
 
#include "stm32f10x.h"

//-------------------------------------------------------
// 与按键状态和动作有关的宏定义
//-------------------------------------------------------
#define  KEY_DOWN		0
#define  KEY_UP			1

#define  KEY1_DOWN		10
#define  KEY1_UP		11
#define  KEY1_DOUBLE	12
#define  KEY1_DOWNLONG	13

#define  KEY2_DOWN		20
#define  KEY2_UP		21
#define  KEY2_DOUBLE	22
#define  KEY2_DOWNLONG	23

#define  KEY3_DOWN		30
#define  KEY3_UP		31
#define  KEY3_DOUBLE	32
#define  KEY3_DOWNLONG	33

#define  KEY4_DOWN		40
#define  KEY4_UP		41
#define  KEY4_DOUBLE	42
#define  KEY4_DOWNLONG	43

#define  KEY_NONE		255

#define  KEYDOWN_LONG_TIME	200		//长按标准,单位为10ms

//-------------------------------------------------------
// 必要的宏定义
//-------------------------------------------------------
#define  KEY1_PIN     GPIO_Pin_13
#define  KEY2_PIN     GPIO_Pin_11
#define  KEY3_PIN     GPIO_Pin_12
#define  KEY4_PIN     GPIO_Pin_2
 
//-------------------------------------------------------
// 库函数操作宏定义
//-------------------------------------------------------
#define  READ_KEY1	GPIO_ReadInputDataBit(GPIOC, KEY1_PIN)
#define  READ_KEY2	GPIO_ReadInputDataBit(GPIOC, KEY2_PIN)
#define  READ_KEY3	GPIO_ReadInputDataBit(GPIOC, KEY3_PIN)
#define  READ_KEY4	GPIO_ReadInputDataBit(GPIOD, KEY4_PIN)
 
//--------------------------------------------------------
// 函数声明
//--------------------------------------------------------
void Key_Init(void);	//按键初始化函数
u8 Key_Scan(void);	    //按键扫描函数	
 
#endif
 

2. key.c文件源码

        我们主要对该文件中的 Key_Scan() 函数源码进行剖析,如代码清单5所示,请大家结合上面的编程要点来阅读源码。

/**
************************************************************
* 代码清单5:在key.c中补充的按键扫描函数
* 函数名称:Key_Scan
* 函数功能:按键IO口扫描
* 入口参数:无
* 返回参数:单击/双击/长按/无动作对应编号
* 说    明:
************************************************************
*/
u8 Key_Scan(void)
{
	u8 downCount = 0, upCount = 0;	//按下和松开的计时变量,单位10ms
	u8 clickDoubleFlag = 0;			//单/双击标志,0为单击,1为双击

	/*------------------- 以下是检测过程 ----------------------*/
	if(READ_KEY1 == KEY_DOWN)
	{
		delay_ms(10);			    //10ms消抖
		if(READ_KEY1 == KEY_DOWN)	//确认按下,开始检测
		{
			while(READ_KEY1==KEY_DOWN && downCount<KEYDOWN_LONG_TIME)
			{	//提前松开按键或按下超过规定时长都会退出循环
				downCount++;	//按下期间计数值累加
				delay_ms(10);	//累加1次的时长
			}
			if(downCount>=KEYDOWN_LONG_TIME)	//说明前面是长按导致的退出
			{
				while(READ_KEY1 == KEY_DOWN);	//等待按键松开
				return KEY1_DOWNLONG;		    //返回长按结果
			}
			else	//说明前面是短按导致的退出,则进行单/双击检测
			{
				for(upCount=0; upCount<25; upCount++)
				{	//在松开期间检测是否有二次按下
					delay_ms(10);
					if(READ_KEY1 == KEY_DOWN)	//有二次按下
					{
						clickDoubleFlag = 1;		    //置双击标志
						while(READ_KEY1 == KEY_DOWN);	//等待按键松开
						return KEY1_DOUBLE;			    //返回双击结果
					}
				}
				if(clickDoubleFlag == 0)	//退出循环发现仍为单击标志
					return KEY1_DOWN;		//返回单击结果
			}
		}
	}
		
	/*---------------------- 扫描按键2/3/4的过程同上 ----------------------*/
	
	return KEY_NONE;	//无按下动作或扫描完成,返回无动作结果
}

        以上代码给出了扫描KEY1按键的过程,由于判断的条件较多且相互嵌套,理解起来是有一点难度的,大家阅读时可以借助Keil的代码收缩和展开的功能(如图8所示),先理清总体上的逻辑关系,再逐层展开仔细阅读,体会编程思路在代码层面上的实现。

图8 Keil中的代码收缩和展开

3. main.c文件源码

        主程序比较简单,如代码清单6所示,主循环中不断根据按键扫描函数的返回值来控制LED的亮灭。单击KEY1,改变红灯状态;双击KEY1,改变绿灯状态;长按KEY1,改变黄灯状态。

/**
  ******************************************************************************
  * 代码清单7:main.c
  * 应用:按键单击/双击/长按控制LED
  * 平台:麒麟座开发板V3.2
  * 作者:老耿
  * 日期:yyyy-mm-dd
  * 修改:无
  ******************************************************************************
  */  

//必要的头文件
#include "delay.h"
#include "key.h"
#include "led.h"

int main()
{
	delay_init();
	LED_Init();
	Key_Init();
	
	while(1)
	{
		switch(Key_Scan())
		{
			case KEY1_DOWN:     RED_TOG();    break;
			case KEY1_DOUBLE:   GREEN_TOG();  break;
			case KEY1_DOWNLONG: YELLOW_TOG(); break;
			default: break;
		}
	}
}

四、再谈延时和消抖方法

        上面的两个实验我们都是用简单的延时实现了按键的消抖。对于这种很简单的演示程序,这样写没问题,但是在实际做项目开发时,程序量往往很大,各种状态值也很多,while(1)主循环要不停的扫描各种状态值是否发生变化,及时的进行任务调度。如果程序中加了这种delay延时操作,则很可能有一件事发生了,但程序还在进行delay延时操作中,而delay结束再去检查那件事的时候,已经晚了。

        为了避免这种情况的发生,要尽量缩短while(1)循环一次所用的时间,而需要进行长时间延时的操作,必须用其他办法来处理,比如通过IO口的外部中断机制或交给专门的定时器去扫描,这些内容我们将在后续的章节讲到。这里只是先给大家强调一种编程的意识,不光是用来消抖的延时,其它任务的延时亦是如此。

(第二部分完,本文结束)

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

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

相关文章

【图论 树 深度优先搜索】2246. 相邻字符不同的最长路径

本文涉及知识点 图论 树 图论知识汇总 深度优先搜索汇总 LeetCode 2246. 相邻字符不同的最长路径 给你一棵 树&#xff08;即一个连通、无向、无环图&#xff09;&#xff0c;根节点是节点 0 &#xff0c;这棵树由编号从 0 到 n - 1 的 n 个节点组成。用下标从 0 开始、长度…

ElementUI样式优化:el-input修改样式、el-table 修改表头样式、斑马格样式、修改滚动条样式、

效果图&#xff1a; 1、改变日期时间组件的字体颜色背景 .form-class ::v-deep .el-date-editor { border: 1px solid #326AFF; background: #04308D !important; } .form-class ::v-deep .el-date-editor.el-input__wrapper { box-shadow: 0 0 0 0px #326AFF inset; } // 输入…

Python | Leetcode Python题解之第203题移除链表元素

题目&#xff1a; 题解&#xff1a; # Definition for singly-linked list. # class ListNode: # def __init__(self, val0, nextNone): # self.val val # self.next next class Solution:def removeElements(self, head: ListNode, val: int) -> Li…

Unity保存玩家的数据到文件中(Unity的二进制序列化)

文章目录 文章运行环境什么是二进制序列化读写文件构造函数 自定义二进制序列化 文章运行环境 Unity2022 什么是二进制序列化 Unity中的二进制序列化是一种将游戏对象或数据结构转换为二进制格式的过程&#xff0c;以便于存储或网络传输。这使数据能够以高效的方式保存&…

网络==>总论v4

既然是写ICT方面的文章&#xff0c;就要不断更新版本&#xff0c;不是文学&#xff0c;可以一劳永逸&#xff0c;如果不更新&#xff0c;看十年前或者二十年前的书意义不大&#xff0c;这就是为啥看到很多编程书都更新到第十几版了&#xff0c;因为要与时俱进。 在去一个地方旅…

JS(JavaScript)事件处理(事件绑定)

天行健&#xff0c;君子以自强不息&#xff1b;地势坤&#xff0c;君子以厚德载物。 每个人都有惰性&#xff0c;但不断学习是好好生活的根本&#xff0c;共勉&#xff01; 文章均为学习整理笔记&#xff0c;分享记录为主&#xff0c;如有错误请指正&#xff0c;共同学习进步。…

C语言单链表的算法之删除节点

一&#xff1a;为什么要删除节点 &#xff08;1&#xff09;有时候链表中节点的数据不想要了&#xff0c;就要删除这个节点 二&#xff1a;删除节点的两个步骤 &#xff08;1&#xff09;第一步&#xff1a;找到这个节点 &#xff08;2&#xff09;第二步&#xff1a;删除这个…

麒麟桌面系统CVE-2024-1086漏洞修复

原文链接&#xff1a;麒麟桌面操作系统上CVE-2024-1086漏洞修复 Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇在麒麟桌面操作系统上修复CVE-2024-1086漏洞的文章。漏洞CVE-2024-1086是一个新的安全漏洞&#xff0c;如果不及时修复&#xff0c;可能会对系统造成安全…

Rust单元测试、集成测试

单元测试、集成测试 在了解了如何在 Rust 中写测试用例后&#xff0c;本章节我们将学习如何实现单元测试、集成测试&#xff0c;其实它们用到的技术还是上一章节中的测试技术&#xff0c;只不过对如何组织测试代码提出了新的要求。 单元测试 单元测试目标是测试某一个代码单…

算法金 | K-均值、层次、DBSCAN聚类方法解析

大侠幸会&#xff0c;在下全网同名「算法金」 0 基础转 AI 上岸&#xff0c;多个算法赛 Top 「日更万日&#xff0c;让更多人享受智能乐趣」 聚类分析概述 聚类分析的定义与意义 聚类分析&#xff08;Clustering Analysis&#xff09;是一种将数据对象分成多个簇&#xff08;…

AI 音乐生成器 MusicGPT,同声传译StreamSpeech!Web短视频平台Sharine

AI 音乐生成器 MusicGPT&#xff0c;同声传译StreamSpeech!Web短视频平台Sharine。 项目简介 MusicGPT 是一款应用程序&#xff0c;允许在任何平台上以高性能方式本地运行最新的音乐生成 AI 模型&#xff0c;而无需安装 Python 或机器学习框架等严重依赖项。 目前它仅支持 Me…

【python】OpenCV—Aruco

文章目录 Detect ArucoGuess Aruco Type Detect Aruco 学习参考来自&#xff1a;OpenCV基础&#xff08;19&#xff09;使用 OpenCV 和 Python 检测 ArUco 标记 更多使用细节可以参考&#xff1a;【python】OpenCV—Color Correction 源码&#xff1a; 链接&#xff1a;http…

二进制方式部署consul单机版

1.consul的下载 mkdir -p /root/consul/data && cd /root/consul wget https://releases.hashicorp.com/consul/1.18.0/consul_1.18.0_linux_amd64.zip unzip consul_1.18.0_linux_amd64.zip mv consul /usr/local/bin/ 2.配置文件 // 配置文件路径&#xff1a; /roo…

Vue 项目运行时,报错Error: Cannot find module ‘node:path‘

Vue 项目运行时&#xff0c;报错Error: Cannot find module ‘node:path’ internal/modules/cjs/loader.js:883throw err;^Error: Cannot find module node:path Require stack: - D:\nodejs\node_modules\npm\node_modules\node_modules\npm\lib\cli.js - D:\nodejs\node_mo…

240628_昇思学习打卡-Day10-SSD目标检测

240628_昇思学习打卡-Day10-SSD目标检测 今天我们来看SSD&#xff08;Single Shot MultiBox Detector&#xff09;算法&#xff0c;SSD是发布于2016年的一种目标检测算法&#xff0c;使用的是one-stage目标检测网络&#xff0c;意思就是说它只需要一步&#xff0c;就能把目标检…

国内多个库被 rsc 钉上 Go 耻辱柱。。。

大家好&#xff0c;我是煎鱼。 这还是比较突然的&#xff0c;下午正努力打工。国内社区群里突然就闹腾起来了。 仔细一看&#xff0c;原来是 Go 核心团队负责人 rsc&#xff0c;又冷不丁搞大招 &#x1f605;。他直接把国内好几个知名库给直接钉上了 Go 源码库的耻辱柱上了。 如…

普元EOS学习笔记-EOS8.3精简版安装

前言 普元EOS是优秀的高低开结合的企业应用软件开发框架。 普元&#xff1a;这是普元公司的名字&#xff0c;普元信息&#xff0c;上市公司哦&#xff0c;这里就不过多安利了。 EOS&#xff1a;普元公司的企业应用开发系统&#xff0c;开发语言是基于Java&#xff0c;技术框…

JavaScript(2)——输入输出和执行顺序

目录 JS的输入输出语法 输出&#xff1a; 输入 JS的代码执行顺序 字面量 JS的输入输出语法 输出&#xff1a; document.write(内容)alert(内容) 页面弹出警告框console.log(内容) 控制台输出语法&#xff0c;程序员调试使用 作用&#xff1a;向body输出内容 注意&…

zabbix-server的搭建

zabbix-server的搭建 部署 zabbix 服务端(192.168.99.180) rpm -ivh https://mirrors.aliyun.com/zabbix/zabbix/5.0/rhel/7/x86_64/zabbix-release-5.0-1.el7.noarch.rpm cd /etc/yum.repos.d sed -i s#http://repo.zabbix.com#https://mirrors.aliyun.com/zabbix# zabbix.r…

音频分离人声和伴奏可以实现吗?手机人声分离工具10款无偿分享!

随着科技的飞速发展&#xff0c;音频处理技术已经取得了显著的进步&#xff0c;其中音频分离人声和伴奏已成为现实。这一技术不仅为音乐制作人和音频工程师提供了便利&#xff0c;更为广大音乐爱好者提供了无限的创作可能性。本文将为大家分享10款手机人声分离工具&#xff0c;…