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

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

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

【学习目标】

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

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

一、认识按键开关

1.1 按键开关的形态和结构

        按键开关主要是指轻触式按键开关,广泛用于各种电器和电子消费品中,有各种各样不同的形态,如图1所示,用圈标注的是我们开发板使用的类型。不管何种形态,它都是一种电子开关,使用时以满足操作力的条件向开关操作方向施压,开关可以闭合接通,当撤销压力时开关即断开,其内部结构是靠金属弹片受力变化来实现通断的。

图1 不同形态的按键开关

        如图2所示,我们的开发板上一共有4个按键开关,用KEYx来表示。这种开关虽然有四个引脚,但我们来看图3,内部①脚和②脚是常闭的,③脚和④脚也是常闭的,这样其实相当于只有两个引脚了。

图2 开发板上的四个按键
图3 四脚按键开关的内部结构示意

1.2 按键的抖动和消抖

        按键机械触点断开或闭合时,由于触点的弹性作用,按键开关不会马上稳定接通或一下子断开,使用按键时会产生带波纹信号,如图4所示,所以需要对波纹信号进行消抖处理,否则可能会引起误判。

图4 按键接通或断开瞬间的抖动纹波

 

        显然上图中的纹波是我们不希望的,因此就需要想办法消抖。消抖的办法有两种:硬件消抖和软件消抖,硬件消抖是在按键两端并联一个0.1uF的电容,利用电容充放电的延时,消除波纹。硬件消抖很少采用,更普遍的做法是通过软件消抖,通过延时10~20ms的方式避开抖动,也可以采用连续多次检测电平的方式避开抖动。

二、按键控制LED编程实践

2.1 任务描述

        本实验的任务是用一个按键实现对一个LED的控制,每按一次,LED的状态就改变一次。在控制方式上,使用了无锁存和有锁存两种方法,分别实现了按下有效和松开有效。

2.2 硬件电路

        图5是开发板上四个按键与STM32连接的电路原理图,连接方式都是一样的,这里就以SW1(KEY1)为例进行介绍。当SW1按下时,KEY1端(PC13)与GND接通,为低电平;当SW1松开时,KEY1端电平被上拉电阻R48拉高。这样,我们通过检测PC13的输入电平就知道KEY1是按下还是松开了。同理,KEY2、KEY3、KEY4的状态需要分别检测PC11、PC12、PD2的输入电平。

图5 四个按键的原理图

2.3 工程文件清单

        本实验的工程文件清单如图6所示,在HARDWARE目录中添加了一对按键的驱动文件 key.ckey.h。由于用到了LED,因此之前的 led.cled.h 需要保留。

图6 按键工程文件清单

2.4 工程代码剖析

1. key.h文件源码

        头文件里依然是与IO有关的宏和函数声明,如代码清单1所示。

 

//-------------------------------------------------------
// 代码清单1:key.h
//-------------------------------------------------------

#ifndef  _KEY_H_
#define  _KEY_H_

#include "stm32f10x.h"

//-------------------------------------------------------
// 必要的宏定义
//-------------------------------------------------------
#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);	//按键初始化函数

#endif

        这里,我们用 GPIO_ReadInputDataBit() 这个库函数来读取一个IO口的电平,函数名虽然长了点,但确实见名就能知意。它有两个参数:GPIOx和GPIO_Pin_x,返回值就是读到的电平(1或0),确实也很直观。

2. key.c文件源码

        该文件里就一个函数 Key_Init(),用来初始化按键的IO口,如代码清单2所示。

/*
 ************************************************************************
 * 代码清单2:key.c
 * 描    述:按键的初始化、驱动
 * 平    台:麒麟座V3.2
 * 作    者:老耿
 * 日    期:yyyy-mm-dd
 * 固 件 库:ST3.5.0
 * 版    本:V1.0
 * 修改记录:无
 ************************************************************************
*/

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

/**
 ************************************************************************
 * 函 数 名:Key_Init
 * 功    能:按键端口初始化
 * 入口参数:无
 * 出口参数:无
 * 说    明:将按键端口设置成输入模式
 ************************************************************************
**/
void Key_Init(void)
{
	//定义一个GPIO初始化对象(结构体)
	GPIO_InitTypeDef  gpio_initstruct;
	
	//打开必要的外设时钟
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC, ENABLE);
	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOD, ENABLE);
	
	//填充初始化结构体,上拉输入模式,并执行生效
	gpio_initstruct.GPIO_Pin = KEY1_PIN | KEY2_PIN | KEY3_PIN;
	gpio_initstruct.GPIO_Mode = GPIO_Mode_IPU;
	//输入模式不用配置Speed参数
	//gpio_initstruct.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init(GPIOC, &gpio_initstruct);
	
	//KEY4与KEY1/2/3不是一组端口,单独再初始化
	gpio_initstruct.GPIO_Pin = KEY4_PIN;
	GPIO_Init(GPIOD, &gpio_initstruct);
}

        虽然本实验只使用到了KEY1,但在代码中,我们将KEY2、KEY3、KEY4的IO口也一并进行了初始化。需要注意的是,GPIO要配置为输入模式。由于按键松开时对应的引脚为高电平,所以我们将其配置为上拉输入模式。

        那为什么没有配置为下拉输入模式呢?那是因为下拉模式,引脚默认为低电平,这相当于按键被按下的状态。这样的话,程序将无法检测按键是否被按下,因为无论按键是否被按下,引脚电平都为低电平了。当然,配置成浮空输入模式当然也是可以的。

        还有一点大家是否发现,上面代码中,没有配置引脚的速度,那是因为引脚处于输入模式时,并不需要配置引脚速度。引脚速度仅是指单片机向引脚刷新电平的频率,所以只有在引脚处于输出模式时才有效。

3. main.c文件源码

        主程序里编写了无锁存和有锁存两种按键控灯的方法,如代码清单3所示,大家测试的时候请保留一种方式的代码,注释掉另一种,来体会这两种方式的差异。

/**
 ******************************************************
 * 代码清单3:main.c
 * 项    目:按键控制LED
 * 任务描述:按一次KEY1,变一次LED
 * 实验平台:OneNET STM32开发板V3.2
 * 作    者:老耿
 * 日    期:yyyy/mm/dd
 ******************************************************
**/
 
//-----------------------------------------------------
// 必要的头文件
//-----------------------------------------------------
#include "delay.h"
#include "led.h"
#include "key.h"


//-----------------------------------------------------
// 主函数
//-----------------------------------------------------
int main()
{
	delay_init();	//延时初始化
	Led_Init();		//LED初始化
	Key_Init();		//按键初始化
	
    while(1)    //以下两种方式保留一种,注释掉另一种
    {
        /*-------------- 无锁存方式 -------------*/
//      if(!READ_KEY1)  //按住红灯亮
//          RED_ON;
//      else
//          RED_OFF;    //松开红灯灭

        /*-------------- 有锁存方式 -------------*/
        if(!READ_KEY1)      //按下KEY1
        {
            delay_ms(10);   //延时消抖
            if(!READ_KEY1)  //确认按下
            {
                while(!READ_KEY1);  //等待松开
                RED_TOG;    //红灯变化
            }
        }
    }
}

2.5 验证与测试

        我们对两种方式分别下载和测试后,大家应该可以发现,无锁存方式下需要按住不放,LED才亮着,一旦松开,LED就灭了。而有锁存的方式下,需要完成按下和松开这一组动作(即单击)才能改变LED的状态,实现这个效果的就是代码清单3中的第36行,当按键被按下时,while的条件将永远成立,这样将导致程序一直停留在此处,只有按键松开时,后续代码才可以得到执行,也就达到了通过按键锁存程序状态的目的。

        此外,在有锁存方式中,当按键被按下后,并不是马上进行后续动作,而是延时了10ms,再次判断按键是否被按下,这样就避免了按键因电平抖动造成被误判的可能。大家也可以尝试去掉消抖这条语句,看每次按键按下的操作是否都能有效。

(第一部分完,共两部分)

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

ELK集群设置密码

一、软件安装清单 elasticsearch7.17.22logstash7.17.22kibana:7.17.22filebeat7.17.22elasticsearch-head:5 二、配置 生成证书 进入elasticsearch容器 bin/elasticsearch-certutil cert -out /usr/share/elasticsearch/config/elastic-certificates.p12 -pass将证书拷贝…

AI职场调研 - 被AI替代的工作分析报告

研究背景 随着人工智能(AI)技术的快速发展,其在职场中的应用日益广泛,引发了对工作被AI替代的担忧。本研究旨在分析在自由职业市场中,哪些工作更有可能被AI替代,并探讨AI对工作市场的实际影响。 研究目标 识别最有可能被AI替代…

OAuth2.0 三方登录(Google登录)

一、OAuth2.0流程 (A)客户端向从资源所有者请求授权。(B)客户端收到授权许可,资源所有者给客户端颁发授权许可(比如授权码code)(C)客户端与授权服务器进行身份认证并出示…

docker部署FastDFS整合Springboot

文章目录 1、FastDFS是什么?2、搭建docker环境3、部署fastdfs4、整合springboot5、接口测试参考文章 1、FastDFS是什么? FastDFS是一个开源的轻量级分布式文件系统,它对文件进行管理,功能包括:文件存储、文件同步、文…

网易Filmly网盘影片播放器安卓TV版

我们在观看网盘内的影视时,想要高清/原画质观看视频,甚至倍速功能往往都需要开通网盘会员才可以,否则你只能使用”马赛克”画质观看。 最近网易上线了一款播放器:Filmly ,它支持直连网盘影视资源,可以高速…

栈,ASCII编码

栈 LinkedList stack new LinkedList<>(); int i 0; while (i < s.length()) { char c s.charAt(i); if (c <) {if (stack.isEmpty()) {i;continue;}stack.removeLast(); //从栈的末尾移除一个元素} else {stack.addLast(c); //压入栈的末尾栈是只允许在一端…

JCR一区级 | Matlab实现BO-Transformer-LSTM多变量回归预测

JCR一区级 | Matlab实现BO-Transformer-LSTM多变量回归预测 目录 JCR一区级 | Matlab实现BO-Transformer-LSTM多变量回归预测效果一览基本介绍程序设计参考资料 效果一览 基本介绍 1.Matlab实现BO-Transformer-LSTM多变量回归预测&#xff0c;贝叶斯优化Transformer结合LSTM长…

论文翻译 | ITER-RETGEN:利用迭代检索生成协同增强检索增强的大型语言模型

论文地址&#xff1a;Enhancing Retrieval-Augmented Large Language Models with Iterative Retrieval-Generation Synergy 摘要 检索增强生成由于有望解决包括过时知识和幻觉在内的大型语言模型的局限性而引起广泛关注。然而&#xff0c;检索器很难捕捉相关性&#xff0c;尤…

ce学习第一天(例行性工作,chrony服务时间同步,两台服务器免密登录)

1、Linux 的例行性工作 1.1单一执行的例行性工作 at 单一执行的例行性工作&#xff1a;单一执行的例行性工作&#xff1a;仅处理执行一次就结束了&#xff0c;at -> atd 1.1.1 at 命令的实际工作过程 1、我们使用 at 命令来生成所要运行的工作&#xff0c;并将这个工作&a…

从中序与后序遍历序列构造二叉树-二叉树题型

106. 从中序与后序遍历序列构造二叉树 - 力扣&#xff08;LeetCode&#xff09; right要再left前面 如下如&#xff0c;后序为第一行&#xff0c;最后一个是根&#xff1b; 中序为第二行&#xff0c;中间的为根&#xff1b; 通过后序的最后一个元素从中序中找到根&#xff0…

935.骑士拨号器 - 力扣

935.骑士拨号器 - 力扣 题目链接&#xff1a;935. 骑士拨号器 - 力扣&#xff08;LeetCode&#xff09; 题目&#xff1a; 示例 1&#xff1a; 输入&#xff1a;n 1 输出&#xff1a;10 解释&#xff1a;我们需要拨一个长度为1的数字&#xff0c;所以把骑士放在10个单元格中…

24/06/26(1.1129)动态内存

strtok 字符串分割函数 #include<stdio.h> int main(){ char str[] "this,a sample string."; char* sep ","; char* pch strtok(str, sep); printf("%s\n", pch); while (pch ! NULL){ printf("%s\…

Power BI 占比函数

1&#xff0c;普通层级结构占比 占比1 DIVIDE([sum_qty], CALCULATE([sum_qty],ALLSELECTED(Item[ITEM_CODE]))) //按照line为一个整理展示数据占比2 SWITCH( true(),ISINSCOPE(Item[ITEM_CODE]),DIVIDE([sum_qty], CALCULATE([sum_qty],ALLSELECTED(Item[ITEM_CODE]))), IS…

说说MQ在你项目中的应用(二)商品支付

看了不少关于MQ的文章&#xff0c;也对MQ的作用做了一些总结。通常来说MQ有三大功能&#xff1a;异步处理、系统解耦和流量削峰。但我觉得这些功能本质上都是围绕着异步这个核心来的&#xff0c;只是针对不同的业务场景做了些调整。 现在市面上常用的MQ中间件&#xff0c;如Ra…

Go语言之函数和方法

个人网站&#xff1a; http://hardyfish.top/ 免费书籍分享&#xff1a; 资料链接&#xff1a;https://url81.ctfile.com/d/57345181-61545511-81795b?p3899 访问密码&#xff1a;3899 免费专栏分享&#xff1a; 资料链接&#xff1a;https://url81.ctfile.com/d/57345181-6…

Java进阶-Lambda

Java进阶-Lambda 前言Lambda表达式什么是Lambda表达式初识Lambda表达式Lambda表达式的简单使用Lambda表达式格式分析与传统接口方法实现的比较 理解Lambda表达式函数式编程非纯函数实例纯函数示例函数式编程在Lambda表达式中的体现 闭包闭包与Lambda表达式的示例 类型推导-匿名…

【D3.js in Action 3 精译】1.2.2 可缩放矢量图形(一)

译注 由于 1.2.2 小节介绍 SVG 的篇幅过多&#xff0c;为了方便查阅&#xff0c;后续将分多个小节依次进行翻译。为了确保整个 1.2.2 小节的完整性&#xff0c;特意将上一篇包含的 SVG 小节的内容整理出来重新编排。敬请留意。 1.2.2 SVG - 可缩放矢量图形 可伸缩矢量图形&…

802.11漫游流程简单解析与笔记_Part2_02_wpa_supplicant、cfg80211、nl80211内核与驱动的关系

wpa、cfg80211、nl80211内核与驱动的关系示意图如下&#xff1a; nl80211和cfg80211都是内核定义的标准接口&#xff0c;目的是规范驱动和应用的统一调用&#xff0c;wpa中常出现nl80211就是通过内核的nl80211接口调用对应cfg80211的部分&#xff0c;进而控制驱动收发数据或切换…

实现高效写入:Schemaless 写入性能优化指南

物联网应用常常需要收集大量的数据&#xff0c;用以支持智能控制、业务分析和设备监控等功能。然而&#xff0c;应用逻辑的更新或硬件的调整可能会导致数据采集项频繁变化&#xff0c;这是时序数据库&#xff08;Time Series Database&#xff0c;TSDB&#xff09;面临的一大挑…

排序算法之java语言实现

零、说在前面 近期打算复习java的几种排序算法&#xff0c;我会将这些排序算法的实现代码、个人心得、时间复杂度分析&#xff0c;算法间的对比做成一个系列帖子&#xff0c;这里作为那些帖子的汇总而存在。 这个系列的框架会包含&#xff1a;概念、实现、时间空间复杂度…