STM32F103中断模式控制led灯亮灭

前言

软件版本

  • STM32CubeMX 6.4 0
  • Keil 531

硬件

  • STM32F103C8T6

1 STM32CubeMX配置项目

1.1 引脚配置

新建项目后,选择STM32F103C8,之后配置引脚

  • A5控制输出灯的亮灭,设置为GPIO_Output
  • A1、A7设置为GPIO_Output,A1持续输出高电平,A7持续输出低电平
  • B5模拟开关,设置为GPIO_EXTI5

配置后引脚如图所示

在这里插入图片描述

1.2 配置EXTI

在NVIC(嵌套向量中断控制器)中,使得PB5能中断

在这里插入图片描述

1.3 配置GPIO

A1设置为高电平,A7为低电平
把B5中断配置为上升沿和下降沿都触发

在这里插入图片描述

1.4 创建项目

完成相关代码配置,创建项目

在这里插入图片描述
在这里插入图片描述

2 Keil配置代码

2.1 查看中断函数

打开生成的项目,找到stm32f1xx_it.c
在这里插入图片描述

找到中断函数EXTI9_5_IRQHandler,选中HAL_GPIO_EXTI_IRQHandler语句,按F12跳到该函数

在这里插入图片描述

在这里插入图片描述

注意:此处跳转必须要运行之后才能到达

2.2 修改代码

往下找到HAL_GPIO_EXTI_Callback这个函数

在这里插入图片描述

该函数是给用户自己重写的,可以在这里根据不同的中断来执行不同的处理。在这里我们需要根据B5的不同中断来实现A5的亮灭。

__weak修饰符:加上了__weak 修饰符的函数,用户可以在用户文件中重新定义一个同名函数,最终编译器编译的时候,会选择用户定义的函数,如果用户没有重新定义这个函数,那么编译器就会执行__weak 声明的函数,并且编译器不会报错。

修改代码如下

    void HAL_GPIO_EXTI_Callback(uint16_t GPIO_Pin){
      if(GPIO_Pin == SWITCH_Pin){
        //获取B5的电位
        GPIO_PinState pinState = HAL_GPIO_ReadPin(SWITCH_GPIO_Port,SWITCH_Pin);

        //低电位
        if(pinState==GPIO_PIN_RESET)
        HAL_GPIO_WritePin(LED_A5_GPIO_Port,LED_A5_Pin,GPIO_PIN_RESET);//把A5变为低电位
        //高电位
        else
            HAL_GPIO_WritePin(LED_A5_GPIO_Port,LED_A5_Pin,GPIO_PIN_SET);//把A4变为高电位
        }
    }

2.3 编译并烧录

通过mcuisp.exe进行烧录

在这里插入图片描述

3 结果

实际运行结果如下

在这里插入图片描述

因为抖动的原因,当B5什么都不接入的时候,灯一直在频闪,看起来是亮的,其实不是很亮;
在接入高电平后LED灯明显变亮;
接入低电平后LED灯熄灭

总结

HAL库可以使用不同方式来完成同一目标,又学习了中断方式控制灯,由于没有开关模块,使用高低电平模拟控制灯的亮灭


参考:
https://blog.csdn.net/as480133937/article/details/98983268
https://blog.csdn.net/qq562029186/article/details/76216311
https://blog.csdn.net/qq_47281915/article/details/121024427?spm=1001.2014.3001.5501

  • 1
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
驱动数码管和旋转编码器代码需要使用 STM32F103C6T6 的引脚和寄存器进行配置和操作。 首先,我们需要确定数码管的引脚连接。假设我们使用了一个 4 位共阳数码管,将其引脚连接到 STM32F103C6T6 的 PB0、PB1、PB2、PB3 引脚上。 下面是使用 C 语言进行驱动数码管的代码: ```c #include "stm32f10x.h" // 定义数码管的段选信号 uint8_t digitSegments[4] = {0x3F, 0x06, 0x5B, 0x4F}; // 数码管显示函数 void displayNumber(uint32_t number) { // 分离个位、十位、百位和千位数字 uint32_t digits[4] = { number % 10, (number % 100) / 10, (number % 1000) / 100, number / 1000 }; // 配置数码管引脚为输出模式 GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStruct); // 逐位显示数字 for (int i = 0; i < 4; i++) { // 点当前位的数码管显示 GPIO_Write(GPIOB, digitSegments[digits[i]] << i); // 添加适当的延时 for (int j = 0; j < 2000; j++); // 关闭数码管显示 GPIO_Write(GPIOB, 0x00); } } int main(void) { // 初始化时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); // 无限循环显示数字 while (1) { for (uint16_t number = 0; number < 10000; number++) { displayNumber(number); } } } ``` 接下来是旋转编码器的代码。假设编码器的 A 和 B 信号连接到了 STM32F103C6T6 的 PA0 和 PA1 引脚上。 ```c #include "stm32f10x.h" // 记录编码器当前状态 uint8_t encoderState = 0; // 编码器状态转换矩阵 const int8_t encoderTable[16] = { 0, +1, -1, 0, -1, 0, 0, +1, +1, 0, 0, -1, 0, -1, +1, 0 }; // 旋转编码器中断处理函数 void EXTI0_IRQHandler(void) { // 清除中断标志 if (EXTI_GetITStatus(EXTI_Line0) != RESET) { EXTI_ClearITPendingBit(EXTI_Line0); } // 读取编码器当前状态 uint8_t currentState = GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_0) | (GPIO_ReadInputDataBit(GPIOA, GPIO_Pin_1) << 1); // 根据编码器状态转换矩阵更新旋转方向 int8_t direction = encoderTable[encoderState * 4 + currentState]; // 更新编码器当前状态 encoderState = (encoderState << 2) | currentState; // 处理旋转方向 if (direction > 0) { // 顺时针旋转 } else if (direction < 0) { // 逆时针旋转 } } int main(void) { // 初始化时钟 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); // 配置编码器引脚为输入模式 GPIO_InitTypeDef GPIO_InitStruct; GPIO_InitStruct.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1; GPIO_InitStruct.GPIO_Mode = GPIO_Mode_IN_FLOATING; GPIO_InitStruct.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA, &GPIO_InitStruct); // 配置外部中断 EXTI_InitTypeDef EXTI_InitStruct; EXTI_InitStruct.EXTI_Line = EXTI_Line0; EXTI_InitStruct.EXTI_Mode = EXTI_Mode_Interrupt; EXTI_InitStruct.EXTI_Trigger = EXTI_Trigger_Rising_Falling; EXTI_InitStruct.EXTI_LineCmd = ENABLE; EXTI_Init(&EXTI_InitStruct); // 配置中断向量表偏移量 NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0); // 配置中断优先级 NVIC_InitTypeDef NVIC_InitStruct; NVIC_InitStruct.NVIC_IRQChannel = EXTI0_IRQn; NVIC_InitStruct.NVIC_IRQChannelPreemptionPriority = 0x00; NVIC_InitStruct.NVIC_IRQChannelSubPriority = 0x00; NVIC_InitStruct.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InitStruct); // 无限循环监听旋转编码器 while (1) { // ... } } ``` 以上是使用 STM32F103C6T6 驱动数码管和旋转编码器的基础代码,具体的细节和功能需求可以根据实际情况进行调整和扩展。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

日常脱发的小迈

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

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

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

打赏作者

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

抵扣说明:

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

余额充值