STM32 4位数码管 无需芯片 显示方法

我使用的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++;
			}		 
		}
	













有啥不懂的欢迎来讨论,虽说我也是小白,所以更欢迎啦!大佬别喷


  • 12
    点赞
  • 56
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值