当前位置:首页 > 单片机 > 单片机
[导读]//======================================================//**基于STM32的按键控制与外部中断实例详解**为了减少重复的内容,所以将外部中断的例程与按键控制实例 放在一起学习。一开始先在51黑论坛率先上传了本文

//======================================================//

**基于STM32的按键控制与外部中断实例详解

**为了减少重复的内容,所以将外部中断的例程与按键控制实例

放在一起学习。

一开始先在51黑论坛率先上传了本文,上传时间不分什么顺序,只是希望在学习中慢慢写下自己的思路。

//======================================================//

/********************************************************/

因为涉及的内容较多,这里大体说一下整片文章的内容分布:

1)、按键控制实例

1、按键的相关信息描述

2、按键的初始化代码实现

3、常用按键检测及控制代码(主要学习其实现思路)

2)、外部中断实例

1、外部中断的相关信息描述

2、外部中断的初始化代码实现

/********************************************************/

==================================华丽的分界线=================================

//=========================按键控制实例=============================//

==================================华丽的分界线=================================

首先,先来讲述简单的按键控制,下面来看看按键是什么东西,开发的时候按键又有哪些要注意的事情.

释义:

按键开关是一种电子开关,属于电子元器件类,使用时以满足操作力的条件向开关操作方向施压开关功能闭合接通,当撤销压力时开关即断开,其内部结构是靠金属弹片受力变化来实现通断的。

一般在开发中涉及按键的一般是按键消抖。按键消抖通常的按键所用开关为机械弹性开关,当机械触点断开、闭合时,由于机械触点的弹性作用,一个按键开关在闭合时不会马上稳定地接通,在断开时也不会一下子断开。也就是在闭合及断开的那一瞬间会有一小段的不稳定状态,即伴随着连续的抖动。按键在电路中一般也就0跟1两中状态,正常情况下,按键没有按下的情况下,按键所连的引脚状态一般都是为1。原因是一般会在引脚上加上一个上拉,然后再与按键相连,按键的另一端接地,所以在按键没有按下的情况下,引脚返回的一般都是高电平,这是电路决定的。如下图:

按键的抖动就是介于1跟0的不定状态。所以为了取得稳定状态,需要在延迟取状态值,也就是既然知道有抖动存在了,那么就把取值的时间延迟一下。还有一个东西跟抖动有关,那就是触发沿的设置,比如上升沿和下降沿的触发。上升沿是按键松开后那一瞬间,电平从0到1的过程,下降沿就与之相反,也就是按键按下的一瞬间,电平从1降为0的过程。边沿触发一般会用在外部中断的过程中。下面的内容也会讲到。

按键相对来说比较简单,按键的初始化跟LED和BEEP蜂鸣器几乎一致,不同之处也就在于按键是输入设备,所以GPIO的模式上一般选为输入模式,紧接着就要设置上拉/下拉输入模式,最后设置初始状态,这个要根据电路来间接。

//=======================key.c===============================//

/************************************************

接口:

key0引脚接在 PE4 低电平有效

key1引脚接在 PE3 低电平有效

key_up引脚接在 PA0 高电平有效

************************************************/

#include "stm32f10x.h"

#include "key.h"

#include "delay.h"

void key_Init(void)

{

RCC ->APB2ENR |= 1 << 2;//使能 PORTA 时钟

RCC ->APB2ENR |= 1 << 6;//使能 PORTE 时钟

//key_up配置

GPIOA ->CRL &= ~(15 << 0);//先设置输入模式

GPIOA ->CRL |= 1 << 3; //PA.0 下拉输入模式

GPIOA ->ODR &= ~(1 << 0); //设置下拉,即设置初始状态为低电平

//key1配置

GPIOE ->CRL &= ~(3 << 13);//先设置输入模式

GPIOE ->CRL |= 1 << 15; //PE.3 上拉输入模式

GPIOE ->ODR |= 1 << 3; //设置上拉,即设置初始状态为高电平

//key0配置

GPIOE ->CRL &= ~(3 << 17);//先设置输入模式

GPIOE ->CRL |= 1 << 19; //PE.4 上拉输入模式

GPIOE ->ODR |= 1 << 4; //设置上拉,即设置初始状态为高电平

}

/*************************************************************

**功 能:按键处理函数

**输入参数:无

**输出参数:

//mode:0,不支持连续按;1,支持连续按;

//0,没有任何按键按下

//1,KEY0 按下

//2,KEY1 按下

//3,KEY_UP 按下 即 WK_UP

**注 释:注意此函数有响应优先级,KEY0>KEY1>KEY_UP!!

***********************************************************/

u8 KEY_Scan(u8 mode)

{

static u8 key_up=1; //按键按松开标志

if(mode)key_up=1; //支持连按

if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))

{

delay_ms(10); //去抖动

key_up=0;

if(KEY0==0)return 1;

else if(KEY1==0)return 2;

else if(WK_UP==1)return 3;

}

else if(KEY0==1&&KEY1==1&&WK_UP==0)key_up=1;

return 0;// 无按键按下

}

//======================================================//

注释:按键检测函数,原理是检测按键的电平,而mode的作用就是一直把key_up置1,那么就会一直进入按键按下的检测,而不会进入按键松开的检测。

if(key_up&&(KEY0==0||KEY1==0||WK_UP==1))这句是进行按键按下的检测,

else if(KEY0==1&&KEY1==1&&WK_UP==0)这句是进行按键松开的检测。

如果没有按键按下的话,虽然key_up为1,但是没有按键,那么就只会返回0。

==================================华丽的分界线=================================

//===========================key.h===========================//

#ifndef __KEY_H_

#define __KEY_H_

#include "stm32f10x.h"

#include "sys.h"

#define KEY0 PEin(4) //PE4

#define KEY1 PEin(3) //PE3

#define WK_UP PAin(0) //PA0 WK_UP 即 KEY_UP

#define KEY0_PRES 1 //KEY0 按下

#define KEY1_PRES 2 //KEY1 按下

#define WKUP_PRES 3 //KEY_UP 按下(即 WK_UP/KEY_UP)

extern void key_Init(void);

extern u8 KEY_Scan(u8 mode);

#endif

//======================================================//

==================================华丽的分界线=================================

//==========================main.c============================//

/*************************************************************

**功 能:按键处理函数

//1,KEY0 按下 控制LED0

//2,KEY1 按下 控制LED1

//3,KEY_UP 按下 即 WK_UP 控制蜂鸣器

***********************************************************/

#include "stm32f10x.h"

#include "sys.h"

#include "delay.h"

#include "led.h"

#include "beep.h"

#include "key.h"

int main(void)

{

u8 key=0;

Stm32_Clock_Init(9); //系统时钟设置

delay_init(72); //延时初始化

led_Init();//LED初始化

beep_Init(); //beep初始化

key_Init(); //key初始化

while(1)

{

key=KEY_Scan(0); //得到键值

if(key)

{

switch(key)

{

case WKUP_PRES: //控制蜂鸣器

GPIOB ->ODR |= (1 << 8);//PB.8 输出高,即蜂鸣器开

delay_ms(10000);//延时

GPIOB ->ODR &= ~(1 << 8);//PB.8 输出低,即蜂鸣器关

break;

case KEY1_PRES: //控制 LED1

GPIOB ->ODR &= ~ (1 << 5);//PB.5输出低电平,即点亮LED

delay_ms(10000); //延时

GPIOB ->ODR |= 1 << 5;//PB.5输出高电平,即灭掉LED

break;

case KEY0_PRES: //控制 LED0

GPIOE ->ODR &= ~(1 << 5);//PB.5输出低电平,即点亮LED

delay_ms(10000); //延时

GPIOE ->ODR |= 1 << 5;//PB.5输出高电平,即灭掉LED

break;

}

}

else delay_ms(10);

}

}

//======================================================//

==================================华丽的分界线=================================

//=========================外部中断实例=============================//

**注释:特别鸣谢CSDN博客的博主->“zzwdkxx”的博文,在外部中断的学习中直接的帮助!

博文地址:http://blog.csdn.net/zzwdkxx/article/details/9036679

==================================华丽的分界线=================================

接着下面将进行外部和中断的讲解,讲之前先对中断先做一定了解。

//============================百度搜索==========================//

简介:

中断是处理器处理外部突发事件的一个重要技术。它能使处理器在运行过程中对外部事件发出的中断请求及时地进行处理,处理完成后又立即返回断点,继续进行处理器原来的工作。引起中断的原因或者说发出中断请求的来源叫做中断源。根据中断源的不同,可以把中断分为硬件中断和软件中断两大类,而硬件中断又可以分为外部中断和内部中断 两类。

外部中断一般是由计算机外设发出的中断请指求,如:键盘中断、打印机中断、定时器中断等。外部中断是可以屏蔽的中断,也就是说,利用中断控制器可以屏蔽这些外部设备 的中断请求。

内部中断是指因硬件出错(如突然掉电、奇偶校验错等)或运算出错(除数为零、运算溢出、单步中断等)所引起的中断。内部中断是不可屏蔽的中断。

软件中断其实并不是真正的中断,它们只是可被调用执行的一般程序。

优先级:

CPU为了处理并发的中断请求,规定了中断的优先权,中断优先权由高到低的顺序是: (1)除法错、溢出中断、软件中断 (2)不可屏蔽

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除( 邮箱:macysun@21ic.com )。
换一批
延伸阅读

嵌入式系统的算法效率与硬件资源的平衡是核心挑战。STM32微控制器通过零开销循环机制与DWT计数器的结合,为算法优化提供了硬件级支持。本文以插入排序算法为例,探讨如何利用STM32的硬件特性验证排序阈值,实现性能与代码复...

关键字: STM32 DWT

智能家居与工业控制场景,手势识别作为非接触式交互的核心技术,正从实验室走向消费级应用。以STM32F407VET6微控制器与TensorFlow Lite Micro框架的组合为例,通过模型量化、硬件加速与低功耗设计,可...

关键字: STM32 TensorFlow

在高性能电机驱动的“纳秒级战争”中,浮点运算单元(FPU)往往成为制约控制环带宽的阿喀琉斯之踵。当PWM载波频率攀升至100kHz,留给电流环PID、Clarke/Park变换及SVPWM计算的时间窗口仅剩寥寥数微秒。此...

关键字: 电机控制算法 FOC磁场 STM32

在工业控制、电机驱动等实时性要求严苛的场景中,中断响应延迟直接影响系统精度与稳定性。STM32系列微控制器凭借Cortex-M内核的硬件特性,通过合理的系统架构设计可实现微秒级中断响应。本文从硬件配置、中断处理、代码优化...

关键字: 裸机开发 STM32

在工业控制、音频处理等高性能嵌入式场景中,某电机驱动项目通过混合使用寄存器操作与CMSIS-DSP库,将PID控制周期从120μs缩短至38μs,系统响应速度提升3倍。本文将揭秘这种"底层+高层"混合编程模式的核心技巧。

关键字: HAL STM32 寄存器

嵌入式系统开发手势识别作为非接触式人机交互的核心技术,正从实验室走向消费级应用。然而,传感器采集的原始信号常因电磁干扰、电源噪声或机械抖动产生失真,导致识别准确率下降。本文以STM32微控制器与PAJ7620手势识别传感...

关键字: STM32 手势识别 噪声

在物联网设备开发中,快速实现稳定可靠的网络通信是项目成功的关键。W5500作为一款集成硬件TCP/IP协议栈的以太网控制器,凭借其"开箱即用"的特性,可大幅缩短STM32平台的网络功能开发周期。本文通...

关键字: STM32 W5500

工业物联网设备开发中,某智能电表项目曾因ADC采样中断响应延迟导致数据丢失率高达15%。技术人员通过重构DMA驱动架构,将数据搬运效率提升12倍,CPU占用率从38%降至3%,成功解决高速采样场景下的实时性难题。这一案例...

关键字: STM32 DMA

工业机器人关节控制系统中,一个典型的伺服驱动器需要在100μs周期内完成电流采样、位置反馈、PID计算和PWM输出等12项关键任务。当传统固定优先级调度导致机械臂出现0.3°的位置抖动时,某运动控制厂商通过引入混合排序算...

关键字: 电机控制 STM32

在STM32嵌入式系统开发中,排序算法的效率直接影响传感器数据处理、通信协议解析等核心任务的实时性。传统快速排序在部分有序数据场景下易退化为O(n²)时间复杂度,而单纯依赖三数取中法优化基准值选择仍存在小规模数据效率不足...

关键字: STM32 传感器
关闭