STM32 使用4个按键控制8位数码管,数码管显示使用定时中断 编程
时间: 2024-05-10 08:16:28 浏览: 99
STM32实现8段数码管动态刷新显示
5星 · 资源好评率100%
首先,你需要连接 4 个按键和 8 位数码管到 STM32 上。然后,你需要编写代码来初始化 GPIO 和定时器,并在定时器中断中更新数码管的显示。
下面是一个简单的例子代码,你可以根据你的需要进行修改:
```c
#include "stm32f10x.h"
#define TIMER_PERIOD 1000 // 定时器中断时间间隔,单位为微秒
#define KEY1_PIN GPIO_Pin_0
#define KEY2_PIN GPIO_Pin_1
#define KEY3_PIN GPIO_Pin_2
#define KEY4_PIN GPIO_Pin_3
#define SEG_A_PIN GPIO_Pin_4
#define SEG_B_PIN GPIO_Pin_5
#define SEG_C_PIN GPIO_Pin_6
#define SEG_D_PIN GPIO_Pin_7
#define SEG_E_PIN GPIO_Pin_8
#define SEG_F_PIN GPIO_Pin_9
#define SEG_G_PIN GPIO_Pin_10
#define SEG_DP_PIN GPIO_Pin_11
volatile uint32_t timer_count = 0;
volatile uint8_t display_value = 0;
void GPIO_Configuration(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
// 初始化按键引脚
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);
GPIO_InitStructure.GPIO_Pin = KEY1_PIN | KEY2_PIN | KEY3_PIN | KEY4_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
GPIO_Init(GPIOA, &GPIO_InitStructure);
// 初始化数码管引脚
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
GPIO_InitStructure.GPIO_Pin = SEG_A_PIN | SEG_B_PIN | SEG_C_PIN | SEG_D_PIN |
SEG_E_PIN | SEG_F_PIN | SEG_G_PIN | SEG_DP_PIN;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init(GPIOB, &GPIO_InitStructure);
}
void TIM2_Configuration(void)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE);
TIM_TimeBaseStructure.TIM_Period = TIMER_PERIOD - 1;
TIM_TimeBaseStructure.TIM_Prescaler = SystemCoreClock / 1000000 - 1;
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1;
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up;
TIM_TimeBaseInit(TIM2, &TIM_TimeBaseStructure);
TIM_ITConfig(TIM2, TIM_IT_Update, ENABLE);
NVIC_InitStructure.NVIC_IRQChannel = TIM2_IRQn;
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0;
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 1;
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
NVIC_Init(&NVIC_InitStructure);
TIM_Cmd(TIM2, ENABLE);
}
void TIM2_IRQHandler(void)
{
if (TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET)
{
TIM_ClearITPendingBit(TIM2, TIM_IT_Update);
timer_count++;
// 每 10 次定时器中断更新一次显示
if (timer_count % 10 == 0)
{
display_value++;
if (display_value > 9)
{
display_value = 0;
}
}
}
}
int main(void)
{
GPIO_Configuration();
TIM2_Configuration();
while (1)
{
// 检测按键状态,根据按键更新显示值
if (GPIO_ReadInputDataBit(GPIOA, KEY1_PIN) == RESET)
{
display_value = 0;
}
if (GPIO_ReadInputDataBit(GPIOA, KEY2_PIN) == RESET)
{
display_value = 1;
}
if (GPIO_ReadInputDataBit(GPIOA, KEY3_PIN) == RESET)
{
display_value = 2;
}
if (GPIO_ReadInputDataBit(GPIOA, KEY4_PIN) == RESET)
{
display_value = 3;
}
// 根据显示值更新数码管显示
switch (display_value)
{
case 0:
GPIO_SetBits(GPIOB, SEG_A_PIN | SEG_B_PIN | SEG_C_PIN | SEG_D_PIN |
SEG_E_PIN | SEG_F_PIN);
GPIO_ResetBits(GPIOB, SEG_G_PIN | SEG_DP_PIN);
break;
case 1:
GPIO_SetBits(GPIOB, SEG_B_PIN | SEG_C_PIN);
GPIO_ResetBits(GPIOB, SEG_A_PIN | SEG_D_PIN | SEG_E_PIN |
SEG_F_PIN | SEG_G_PIN | SEG_DP_PIN);
break;
case 2:
GPIO_SetBits(GPIOB, SEG_A_PIN | SEG_B_PIN | SEG_D_PIN | SEG_E_PIN |
SEG_G_PIN);
GPIO_ResetBits(GPIOB, SEG_C_PIN | SEG_F_PIN | SEG_DP_PIN);
break;
case 3:
GPIO_SetBits(GPIOB, SEG_A_PIN | SEG_B_PIN | SEG_C_PIN | SEG_D_PIN |
SEG_G_PIN);
GPIO_ResetBits(GPIOB, SEG_E_PIN | SEG_F_PIN | SEG_DP_PIN);
break;
}
// 延时等待一段时间,使数码管显示稳定
for (int i = 0; i < 10000; i++);
}
}
```
在这个例子中,我们使用 TIM2 定时器,并将其配置为每隔 1 毫秒触发一次中断。在中断处理函数中,我们使用一个计数器变量 `timer_count` 记录定时器中断的次数,并在每 10 次中断时更新一次数码管的显示值 `display_value`。
在 `main` 函数中,我们不断循环检测按键状态和更新数码管的显示。根据按键状态更新 `display_value`,然后根据 `display_value` 更新数码管的显示。为了使数码管显示稳定,我们在更新显示后添加了一个简单的延时。
当程序运行时,你应该能够看到数码管上显示的数字随着按键的按下而改变,同时每隔一段时间数码管上的数字也会自动更新。
阅读全文