【STM32+HAL库】---- 按键中断控制LED

硬件开发板:STM32G0B1RET6
软件平台:cubemax+keil+VScode

1 新建cubemax工程

1.1 配置系统时钟树

image

1.2 配置相关GPIO引脚

①LED由PC13引脚控制
image
选择PA5引脚,GPIO_Output模式
image
GPIO模式配置:
image

②按键开关由PC13引脚控制
image
选择PC13引脚,GPIO_EXTIx模式,其中13表示13号中断线
image
GPIO模式配置:按键设置为下降沿触发中断(LED)
image

1.3 配置NVIC中断

勾上Enabled (G0系列相比于F1系列没有优先级Group配置)
image

1.4 导出工程

…略

2 中断处理流程

① main.c 中的 MX_GPIO_Init() 函数调用HAL库里的使能中断函数和设置中断优先级函数
image

② 在 HAL_NVIC_EnableIRQ() 中调用 NVIC_EnableIRQ()
image

③ 当触发中断时,处理器查表跳转到位于 stm32g0xx.it.c 里的中断服务程序 EXTIx_IRQHandler(void) 【这里的x指中断线】;中断服务程序 EXTIx_IRQHandler (void) 中调用外部中断通用处理函数HAL_GPIO_EXTI_IRQHandler (uint16_t GPIO_Pin) ,该函数作为HAL库提供的外部中断接口函数,参量为触发中断的引脚
image

④相比较于F系列,G0系列的HAL_GPIO_EXTI_IRQHandler (uint16_t GPIO_Pin)里所调用的中断回调函数分成了上升沿和下降沿两部分
G0系列:
image

F1/F4系列:
image

⑤在stm32g0xx.it.c 中重写对应类型的中断回调函数,并补全中断所需要执行的任务函数

3 代码

重写下降沿中断回调函数

//微秒级的延时
void delay_us(uint32_t delay_us)
{
  volatile unsigned int num;
  volatile unsigned int t;
  for (num = 0; num < delay_us; num++)
  {
    t = 11;
    while (t != 0)
    {
      t--;
    }
  }
}

//毫秒级的延时
void delay_ms(uint16_t delay_ms)
{
  volatile unsigned int num;
  for (num = 0; num < delay_ms; num++)
  {
    delay_us(1000);
  }
}

//下降沿回调函数
void HAL_GPIO_EXTI_Falling_Callback(uint16_t GPIO_Pin)
{
  if (GPIO_Pin==BUTTON_Pin)
  {
    delay_ms(20);     /*消抖*/
    if (GPIO_Pin==BUTTON_Pin)
    {
      HAL_GPIO_TogglePin(LED_GPIO_Port,LED_Pin);
      while (BUTTON_Pin==0);
    }
  }
}
### 使用HALSTM32上通过按键中断控制LED灯 #### 配置环境与工具链 为了实现这一目标,首先需要安装并配置好必要的开发环境。这包括但不限于安装STM32CubeMX以及相应的IDE(如Keil MDK或TrueSTUDIO)。确保已经正确设置了编译器路径和其他必要选项。 #### STM32CubeMX配置步骤 1. 打开STM32CubeMX软件,创建新项目并选择合适的MCU型号,这里假设使用的是STM32F407。 2. 进入Pinout & Configuration界面,在RCC部分启用HSE作为系统时钟源以获得更高的性能[^1]。 3. 对于GPIO配置: - 将PF9配置为推挽输出模式用于驱动LED- 设置PC13或其他适合的引脚为EXTI Line Input Pull-up/down模式连接至按键开关,并关联到对应的外部中断线(External Interrupt line)[^2]。 4. 在NVIC (Nested Vectored Interrupt Controller)设置里激活所需的外部中断通道优先级。 5. 利用中间件(Middleware)/RTOS/其他组件根据实际需求进行额外设定。 6. 完成上述操作后点击“Project”菜单下的“Generate Code”,让STM32CubeMX自动生成初始化代码框架。 #### 编写应用程序逻辑 接下来是在生成的基础之上编写具体的业务逻辑: ```c #include "main.h" // 声明全局变量 extern UART_HandleTypeDef huart1; uint8_t key_pressed_flag = 0; void SystemClock_Config(void); static void MX_GPIO_Init(void); int main(void){ HAL_Init(); SystemClock_Config(); // 初始化系统时钟 MX_GPIO_Init(); while (1){ if(key_pressed_flag == 1){ // 如果检测到了按键按下事件,则切换LED状态 HAL_GPIO_TogglePin(GPIOF, GPIO_PIN_9); // 控制PF9上的LED亮起或熄灭 key_pressed_flag = 0; // 清除标志位 HAL_Delay(200); // 添加短延时防止抖动影响 } /* USER CODE END WHILE */ /* USER CODE BEGIN 3 */ } } /** * @brief EXTI line detection callbacks. */ void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){ if(GPIO_Pin == GPIO_PIN_13){ key_pressed_flag = 1; // 当检测到按键触发时设置标志位 } } ``` 这段代码实现了当用户按下了连接到PC13的按钮之后,会立即改变PF9端口的状态从而达到点亮或者关闭LED的效果。同时为了避免因机械触点反弹造成误判的情况发生,在处理函数内部加入了简单的去抖机制——即每次执行完一次完整的动作序列之前都会等待一段时间再继续监听新的输入信号到来。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

@Luminescence

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值