STM32学习笔记(三)按键点亮LED灯(中断法)

一、操作思路

中断(Interrupt)模式:

  • 为了提高CPU的效率和使系统具有良好的实时性,可以采用中断控制I/O方式。采用中断方式CPU就不必花费大量时间去查询各外围设备的状态了。而是当外围设备需要请求服务时,向CPU发出中断请求(ARQ),CPU响应外围设备中断,停止执行当前程序,转去执行一个外围设备服务的程序,此服务程序称为中断服务处理程序,或称中断服务子程序。中断处理完毕,CPU又返回来执行原来的程序。

  • 输入(按键):
    KEY1:PA0
    KEY2:PA1

  • 输出(LED灯):
    LED1:PB8
    LED2:PB9

步骤:
1.配置时钟
2.配置GPIO口
3.使能中断
4.配置工程

二、主函数main

#include "main.h"
#include "gpio.h"
 
void SystemClock_Config(void);  //因为这个函数在main函数中,如果想要使用这个函数,需要提前声明
 
int main(void)
{
  HAL_Init();                   //初始化HAL库
  SystemClock_Config();         //初始化系统时钟
  MX_GPIO_Init();               //初始化GPIO引脚口
 
void SystemClock_Config(void)   //配置系统时钟函数
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
 
  /** Initializes the RCC Oscillators according to the specified parameters
  * in the RCC_OscInitTypeDef structure.
  */
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
 
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
 
  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}

三、GPIO代码-----gpio.c

#include "gpio.h"
 
void MX_GPIO_Init(void)         //初始化GPIO引脚函数
{
 
  GPIO_InitTypeDef GPIO_InitStruct = {0};  //用于初始化设置GPIO的结构体
 
  /* GPIO Ports Clock Enable */
  __HAL_RCC_GPIOD_CLK_ENABLE();     //使能GPIOD时钟
  __HAL_RCC_GPIOA_CLK_ENABLE();		//使能GPIOA时钟
  __HAL_RCC_GPIOB_CLK_ENABLE();		//使能GPIOB时钟
 
  /*Configure GPIO pin Output Level */
  HAL_GPIO_WritePin(GPIOB, GPIO_PIN_8|GPIO_PIN_9, GPIO_PIN_SET);  //GPIO引脚输出1或者0的函数
 
  /*Configure GPIO pins : PA0 PA1 */              
  GPIO_InitStruct.Pin = GPIO_PIN_0|GPIO_PIN_1;	  //配置GPIO引脚:PA0 PA1
  GPIO_InitStruct.Mode = GPIO_MODE_IT_FALLING;    //具有下降沿触发检测的外部中断模式
  GPIO_InitStruct.Pull = GPIO_NOPULL;             //内部电阻既不拉高也不拉低
  HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);         //根据上述内容,初始化GPIOA引脚
 
  /*Configure GPIO pins : PB8 PB9 */              
  GPIO_InitStruct.Pin = GPIO_PIN_8|GPIO_PIN_9;	  //配置GPIO引脚:PB8 PB9
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;     //推挽输出;推挽输出可以真正能真正的输出高
  GPIO_InitStruct.Pull = GPIO_NOPULL;             //内部电阻既不拉高也不拉低
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;    //GPIO引脚输出的频率
  HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);         //根据上述内容,初始化GPIOB引脚
 
  /* EXTI interrupt init*/
  HAL_NVIC_SetPriority(EXTI0_IRQn, 0, 0);         //配置EXTI0_IRQn中断,抢先优先级0,响应优先级0
  HAL_NVIC_EnableIRQ(EXTI0_IRQn);                 //使能中断通道EXTI0_IRQn
 
  HAL_NVIC_SetPriority(EXTI1_IRQn, 0, 0);         //配置EXTI1_IRQn中断,抢先优先级0,响应优先级0
  HAL_NVIC_EnableIRQ(EXTI1_IRQn);                 //使能中断通道EXTI1_IRQn
 
}

四、中断服务函数

stm32f1xx_it.h中,按键按下,检测到一个下降沿,就会跳转到这个函数中

#include "main.h"
#include "stm32f1xx_it.h"
 
void EXTI0_IRQHandler(void)                //中断服务函数
{
  HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_0);    //调用中断处理公用函数,根据输入的参数,来判断到底输入的是哪个按键
}
 
void EXTI1_IRQHandler(void)                //中断服务函数
{
  HAL_GPIO_EXTI_IRQHandler(GPIO_PIN_1);    //调用中断处理公用函数,根据输入的参数,来判断到底输入的是哪个按键
}

五、用外部中断共用入口函数,来调用外部中断处理回调函数

外部中断共用入口函数:void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)

void HAL_GPIO_EXTI_IRQHandler(uint16_t GPIO_Pin)
{
  /* EXTI line interrupt detected */
  if (__HAL_GPIO_EXTI_GET_IT(GPIO_Pin) != 0x00u)  //确保是否产生了中断
  {
    __HAL_GPIO_EXTI_CLEAR_IT(GPIO_Pin);           //清除中断标志位
    HAL_GPIO_EXTI_Callback(GPIO_Pin);             //调用中断服务回调函数
  }
}

六、编写外部中断处理回调函数

void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)    //外部中断处理回调函数
{
	switch(GPIO_Pin)
	{
		HAL_Delay(50);
		case GPIO_PIN_0:
			if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==  GPIO_PIN_RESET){   //消抖函数,读取GPIO输入口的引脚是否为低电平
				HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8);   //翻转电平函数,GPIO口的输出口,翻转一次B8的电平
			}
		break;
		case GPIO_PIN_1:
			if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_1)==  GPIO_PIN_RESET){   //消抖函数,读取GPIO输入口的引脚是否为低电平
				HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9);   //翻转电平函数,GPIO口的输出口,翻转一次B9的电平
			}
		break;
	}
}

七、把上述的外部中断处理回调函数,放到main主函数中

#include "main.h"
#include "gpio.h"
 
void SystemClock_Config(void);  //因为这个函数在main函数中,如果想要使用这个函数,需要提前声明
void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin)    //外部中断处理回调函数
{
	switch(GPIO_Pin)
	{
		HAL_Delay(50);    //延时50ms
		case GPIO_PIN_0:
			if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_0)==  GPIO_PIN_RESET){   //消抖函数,读取GPIO输入口的引脚是否为低电平
				HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_8);   //翻转电平函数,GPIO口的输出口,翻转一次B8的电平
			}
		break;
		case GPIO_PIN_1:
			if(HAL_GPIO_ReadPin(GPIOA,GPIO_PIN_1)==  GPIO_PIN_RESET){   //消抖函数,读取GPIO输入口的引脚是否为低电平
				HAL_GPIO_TogglePin(GPIOB,GPIO_PIN_9);   //翻转电平函数,GPIO口的输出口,翻转一次B9的电平
			}
		break;
	}
}
int main(void)
{
  HAL_Init();                   //初始化HAL库
  SystemClock_Config();         //初始化系统时钟
  MX_GPIO_Init();               //初始化GPIO引脚口
}
void SystemClock_Config(void)   //配置系统时钟函数
{
  RCC_OscInitTypeDef RCC_OscInitStruct = {0};
  RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
 
  RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
  RCC_OscInitStruct.HSEState = RCC_HSE_ON;
  RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
  RCC_OscInitStruct.HSIState = RCC_HSI_ON;
  RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
  RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
  RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
  if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
  {
    Error_Handler();
  }
 
  RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
                              |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
  RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
  RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
  RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
  RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
 
  if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
  {
    Error_Handler();
  }
}
  • 0
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
STM32是一款基于ARM Cortex-M系列内核的32位微控制器,可用于开发嵌入式系统。按键中断LED亮灭是嵌入式系统中非常常见的功能,下面我将用300字回答如何使用STM32按键中断控制LED的亮灭。 首先,我们需要连接一个按键一个LEDSTM32微控制器的相应引脚上。假设按键连接到PA0引脚,LED连接到PC13引脚。 接下来,我们需要配置STM32的GPIO外设来控制这些引脚。首先,开启相关引脚的时钟。然后,将PA0引脚配置为输入模式,PC13引脚配置为输出模式。可以使用STM32的寄存器或者开发环境提供的库函数来完成这些配置。 然后,我们需要配置外部中断。配置PA0引脚所对应的外部中断线,使其可以检测到按键的状态变化。可以使用STM32的寄存器或者库函数来完成这个配置。 紧接着,我们编写中断处理函数。当按键状态发生变化,中断触发时,中断处理函数被调用。在该函数中,我们可以读取按键引脚的状态,如果按键被按下,我们将PC13引脚设置为高电平,LED亮起;如果按键被释放,我们将PC13引脚设置为低电平,LED熄灭。 最后,我们需要在主函数中启用中断。启用中断后,当按键状态发生变化时,中断处理函数将被调用。 以上就是使用STM32按键中断来控制LED亮灭的步骤。通过配置GPIO外设和外部中断,编写中断处理函数,我们能够实现按下按键时,LED亮起;释放按键时,LED熄灭的功能。这样就完成了按键中断控制LED亮灭的任务。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值