STM32 数码管显示

配置相应的GPIO口为通用输出口,让数码管循环显示00~FF。

#include "stm32f10x.h"

#include "smg.h"

uint8_t table[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x5e,0x79,0x71};

void delay(int n){

for(int i=0;i<n;i++){

for(int j=0;j<n;j++);

}

}

int main()

{

uint16_t temp=0;

SMG_Init();

while(1){

for(int i=0;i<255;i++){

temp=(table[i/16]<<8)|table[i%16];

GPIO_Write(GPIOC,temp);

delay(500);

}

}

}

smg.c

#include "stm32f10x.h"

#include "smg.h"

void SMG_Init(){

RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE);   //AF_PP复用推挽输出  //Out_OD开漏

GPIO_InitTypeDef GPIO_InitStructure;                     //IPU 上拉输入  //IPD 下拉输入

GPIO_InitStructure.GPIO_Pin =  GPIO_Pin_All;

GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;   //GPIO???

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;  //GPIO推挽输出  //AF_OD复用开漏输出

GPIO_Init(GPIOC,&GPIO_InitStructure);

}

smg.h

#ifndef __SMG_H

#define __SMG_H

void SMG_Init(void);

#endif

<think>好的,用户想用STM32F103VB驱动数码管实现滚动显示“HELLO”的功能。首先,我需要确认数码管的类型,是共阴还是共阳,这会影响电路连接和代码中的电平控制。假设用户使用的是共阴数码管,那么段选信号需要高电平来点亮,位选则低电平选中。 接下来,硬件连接方面,数码管的段选通常需要连接到GPIO的8个引脚,对应a到dp段。位选的话,如果要显示多个字符,可能需要多个位选线,但滚动显示可能只需要一个数码管,通过动态扫描实现滚动效果。不过用户提到滚动显示,可能需要多个数码管或者利用移位寄存器来扩展IO口。不过STM32F103VB的IO资源应该足够,但需要确认具体引脚分配。 然后,代码部分需要初始化GPIO,设置推挽输出模式,速度根据需求选择,比如10MHz。根据参考的引用[^1],PE口的某些引脚可能被占用,所以需要避开这些引脚,选择其他可用的GPIO口,比如PA或PB口。需要检查STM32F103VB的具体引脚功能,确保使用的IO口没有复用功能冲突。 数码管的驱动需要段码表,即每个字符对应的段选值。比如H的段码是0x76,E是0x79,L是0x38,O是0x3F。滚动显示通常通过移位段码数组中的字符,或者改变位选的位置来实现。动态扫描需要定时器中断来刷新显示,防止闪烁。定时器的配置需要设置适当的中断频率,比如1ms一次,每次切换显示的字符位置,形成滚动效果。 用户可能需要示例代码,所以需要编写初始化函数,包括GPIO和定时器的配置。主循环中处理字符数组的移位,定时器中断服务程序中负责刷新数码管显示。需要注意的是,中断服务函数中不能有长时间的操作,以免影响系统性能。 另外,滚动速度的控制可以通过调整字符移位的间隔时间,比如每500ms移动一次。这可以通过另一个定时器或者系统滴答定时器来实现。同时,数码管的消影处理也很重要,即在切换显示时先关闭所有段选,再更新段码和位选,避免残影。 最后,测试代码时需要确保硬件连接正确,特别是限流电阻的使用,防止电流过大损坏数码管或IO口。如果用户遇到问题,可能需要检查电路连接、段码表是否正确,以及定时器配置是否合适。</think>根据STM32F103VB芯片特性及数码管驱动原理,为您提供滚动显示"HELLO"的实现方案: ### 一、硬件配置要点 1. **数码管选择**:推荐使用4位共阴数码管(如5461BS),需8个GPIO控制段选(a-dp),4个GPIO控制位选 2. **限流电阻**:每个段选引脚串联220Ω电阻,防止过流烧毁(最大电流不超过20mA) 3. **GPIO分配**: - 段选:PA0-PA7(对应a-dp) - 位选:PB0-PB3(4位显示) ### 二、软件实现核心代码 ```c // 数码管段码表(共阴0-9 A-F H L O) const u8 SEG_CODE[] = { 0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07, // 0-7 0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71, // 8-F 0x76,0x38,0x3F // H(16), L(17), O(18) }; // 显示缓冲区 u8 disp_buf[5] = {16, 14, 17, 17, 18}; // H E L L O u8 pos = 0; // 显示位置 // GPIO初始化 void Seg_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 段选PA0-PA7 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); GPIO_InitStructure.GPIO_Pin = 0x00FF; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_10MHz; GPIO_Init(GPIOA, &GPIO_InitStructure); // 位选PB0-PB3 RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3; GPIO_Init(GPIOB, &GPIO_InitStructure); } // 定时器中断实现滚动 void TIM2_IRQHandler(void) { static u8 cnt = 0; if(TIM_GetITStatus(TIM2, TIM_IT_Update) != RESET) { // 消隐处理 GPIO_SetBits(GPIOB, GPIO_Pin_0 | GPIO_Pin_1 | GPIO_Pin_2 | GPIO_Pin_3); // 输出段码 GPIOA->ODR = SEG_CODE[disp_buf[pos]] << 8; // 选择位 switch(cnt % 4) { case 0: GPIO_ResetBits(GPIOB, GPIO_Pin_0); break; case 1: GPIO_ResetBits(GPIOB, GPIO_Pin_1); break; case 2: GPIO_ResetBits(GPIOB, GPIO_Pin_2); break; case 3: GPIO_ResetBits(GPIOB, GPIO_Pin_3); break; } // 每20次中断移动一次显示位 if(++cnt >= 20) { cnt = 0; pos = (pos + 1) % 5; // 循环显示H-E-L-L-O } TIM_ClearITPendingBit(TIM2, TIM_IT_Update); } } ``` ### 三、实现步骤 1. **硬件连接**:按上述GPIO分配连接数码管,注意共阴公共端接GND 2. **定时器配置**: - 配置TIM2为1ms周期中断(72MHz主频下,预分频7200-1,重载值10-1) - 使能中断并启动定时器 3. **主函数初始化**: ```c int main(void) { Seg_Init(); TIM2_Config(); // 定时器配置函数 while(1) { // 主循环可添加其他控制逻辑 } } ``` ### 四、效果优化建议 1. **亮度调节**:通过PWM控制位选端的导通时间 2. **滚动速度**:修改定时器中断周期或移动间隔计数(20次判断) 3. **消隐改进**:在切换位选前增加1us延时,消除鬼影现象
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值