我使用的STM32F103C8T6
4位共阳数码管
数码管的多位显示,一开始认为多位显示必须用芯片才可以,后来发现了一个新的思路,就是扫描的方式,逐字快速显示,肉眼无法察觉的频率,让其看似都在同时显示的方式。
开始思路固化,一直在想办法使用同一套IO口,也就是一直是A系列的IO口来控制位选(也就是说让某一位亮),但是用同一组IO的问题就是前面修改位选IO口电平后又被段选修改,无法达到效果,后来突然开窍,为何不用B系列的IO口控制位选呢?
共阳数码管对应的
0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90
共阴极数码管
0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f
什么原理呢?把这些16进制数据转换成2进制在对应下面的表和图,
共阳极也就是说,公共端是输入正极的,某个引脚给低电平,那么那个引脚对应的块就会亮。
比如说想让“1”亮起来,那么就是点亮“B”和“C”的部分,也就是
这样,哦对了注意下,读取的时候是从低位到高位,是倒过来的,不要弄反了。11111001对应的16进制数据就是0xF9 GPIO_Write(GPIOA,0xF9);就会显示“1”啦。
接线看这儿
对应着接线就行,上面的“12”“9”“8”“6”是用来控制每个数码管的通断电用的 我用的B0 B1 B2 B3控制的,下面就是按照从左往右的顺序接线就行,从A0开始。接好之后就是..先看看视频哈哈。代码在下面 别急
这是表面上看起来
实际上是这样一个一个扫描的,只要够快,肉眼就察觉不到,思路有了吧?
这个是动态显示数据,还可以的。
然后就是令人激动人心的代码部分了哈哈哈....记得点赞点赞哦。
注意下面的位选,也是2进制转成16进制的数据,看不懂就去转成2进制看看。
smg.c
#include "smg.h"
#include "stm32f10x.h"
#include "delay.h"
void Led_Code(int date,int delay);
char Code[10]={ 0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};
//显示段码 数码管字型 共阳极0-9
//int wei[6]={0xfe,0xfd,0xff7b,0xff3,0x7f8,0xff0};
int wei[4]={0x1,0x2,0x20,0x40};
//个十百千
//4个 位选端
void SMG_Init(void)
{
GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE); //使能GPIOB时钟
GPIO_InitStructure.GPIO_Pin = 0xffff; //PC0-PC15引脚配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //配置引脚为推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //GPIOB速度为50MHz
GPIO_Init(GPIOA, &GPIO_InitStructure); //初始化PC0-PC15
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); //使能GPIOB时钟
GPIO_InitStructure.GPIO_Pin = 0xffff; //PC0-PC15引脚配置
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //配置引脚为推挽输出
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; //GPIOB速度为50MHz
GPIO_Init(GPIOB, &GPIO_InitStructure); //初始化PC0-PC15
}
void Led_Code(int date,int delay) //显示函数
{
int c,d,e,f;
c=date%10000/1000; //求千位
d=date%1000/100; //求百位
e=date%100/10; //求十位
f=date%10; //求个位
GPIO_Write(GPIOA,0xff);
GPIO_Write(GPIOB,wei[0]);
GPIO_Write(GPIOA,Code[f]);//个
delay_ms(delay);
GPIO_Write(GPIOA,0xff);
GPIO_Write(GPIOB,wei[1]);
GPIO_Write(GPIOA,Code[e]);//十
delay_ms(delay);
GPIO_Write(GPIOA,0xff);
GPIO_Write(GPIOB,wei[2]);
GPIO_Write(GPIOA,Code[d]);//百
delay_ms(delay);
GPIO_Write(GPIOA,0xff);
GPIO_Write(GPIOB,wei[3]);
GPIO_Write(GPIOA,Code[c]);//千
delay_ms(delay);
}
smg.h
#ifndef __ESP_H
#define __ESP_H
void SMG_Init(void);
/* 这里给出两中 参考码,方便取用
unsigned char shu1[]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x40}; //没有dp小数点
//数字0-9的共阴极码表 0 - 9 -(下标10)
unsigned char shu0[]={0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90,0xbf};
//数字0-9的共阳极码表 0 - 9 -(下标10)
*/
//int wei[4]={0x1,0x2,0x20,0x40};
// char Code[10]={ 0xc0,0xf9,0xa4,0xb0,0x99,0x92,0x82,0xf8,0x80,0x90};
void Led_Code(int date,int delay);
#endif
我开了一个定时器,用来显示变化的数据
#include "timer.h"
#include "led.h"
#include "delay.h"
int count=0;
void TIM3_Int_Init(u16 arr,u16 psc)
{
TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure;
NVIC_InitTypeDef NVIC_InitStructure;
RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE);
// RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM2, ENABLE); //时钟使能
//定时器TIM3初始化
TIM_TimeBaseStructure.TIM_Period = arr; //设置在下一个更新事件装入活动的自动重装载寄存器周期的值
TIM_TimeBaseStructure.TIM_Prescaler =psc; //设置用来作为TIMx时钟频率除数的预分频值
TIM_TimeBaseStructure.TIM_ClockDivision = TIM_CKD_DIV1; //设置时钟分割:TDTS = Tck_tim
TIM_TimeBaseStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM向上计数模式
TIM_TimeBaseInit(TIM3, &TIM_TimeBaseStructure); //根据指定的参数初始化TIMx的时间基数单位
TIM_ITConfig(TIM3,TIM_IT_Update,ENABLE ); //使能指定的TIM3中断,允许更新中断
//定时器TIM2初始化
//中断优先级NVIC设置
NVIC_InitStructure.NVIC_IRQChannel = TIM3_IRQn; //TIM3中断
NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority = 0; //先占优先级0级
NVIC_InitStructure.NVIC_IRQChannelSubPriority = 3; //从优先级3级
NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IRQ通道被使能
NVIC_Init(&NVIC_InitStructure); //初始化NVIC寄存器
TIM_Cmd(TIM3, ENABLE); //使能TIMx
//TIM_Cmd(TIM2, ENABLE); //使能TIMx
}
//定时器3中断服务程序
void TIM3_IRQHandler(void) //TIM3中断
{
if (TIM_GetITStatus(TIM3, TIM_IT_Update) != RESET) //检查TIM3更新中断发生与否
{
TIM_ClearITPendingBit(TIM3, TIM_IT_Update ); //清除TIMx更新中断标志
count++;
}
}
有啥不懂的欢迎来讨论,虽说我也是小白,所以更欢迎啦!大佬别喷