STM32学习记录——74HC595四位数码管显示

数模管作为STM32的一个重要外设,由于其成本低、稳定,被用于许多场景中。本篇文章来介绍下四位数码管的使用方法。




一、数码管的分类

数码管分为共阴数码管和共阳数码管,共阴和共阳的区别就在于,输出1还是输出0的时候,能点亮数码管的一小段。

在这篇文章中,所使用的是由两片74HC595芯片控制的4位数码管
在这里插入图片描述

在这里插入图片描述
挂几张图

二、74HC595芯片(串入并出)

该芯片是一个8位串行输入、并行输出的位移缓存器。(缓存器在控制数码管中具有重要作用)
芯片引脚图
16个管脚分别对应的作用Q0~Q7:并行输出
Q7’:串行输出
SH_CP:移位寄存器时钟输入
ST_CP:存储寄存器时钟输入
DS:串行输入

使用两片74HC595芯片,第一片74HC595芯片的Q7口,也就是串行输出口可以向下一片的74HC595芯片的串行输入口输入数据。
第二片74HC595芯片是用于控制数码管输出显示,而第一片74HC595芯片只使用了QA、QB、QC、QD四个管脚来储存信息。

两片74HC595芯片之间的联系

三、原理图

原理图
对于4位的数码管模块,接线选择为

VCC------------>供电
DIO------------>PB13
RCLK------------>PB12  当八个数据完全传入缓冲区,此时拉高RCLK,八位数据便可以一次性全部输出。
SCLK------------>PB15   控制串入的时钟信号线,当有一个上升沿,便传入一个数据。
GND------------>接地

本文主要使用PB13 、PB12 、PB15 来控制数码管的显示数据。
以下使用数码管也是以这三个端口进行分析。

四、代码主要操作

主要是由以下四个函数构成:

void HC595_GPIO_Configuration(void);
void HC595_Send_Data(unsigned char num, unsigned char show_bit);
void HC595_Send_Byte(unsigned char byte)void display(unsigned int n);

五、代码分析

1.void HC595_GPIO_Configuration(void);

该函数主要是使能接在4位数码管的主要GPIO口,如上文提到的一般:

/************************
VCC------------>供电
DIO------------>PB13
RCLK------------>PB12  
SCLK------------>PB15  
GND------------>接地
**************************/

unsigned int num[] = {0xc0, 0xf9, 0xa4, 0xb0, 0x99, 0x92, 0x82, 0xf8, 0x80, 0x90, 0xff, 0x00};
//创建一个数组,0-9所对应的十六进制数

void HC595_GPIO_Configuration(void)
{
	GPIO_InitTypeDef GPIO_InitStructure;

	RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);
	
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12 | GPIO_Pin_13 | GPIO_Pin_15;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
	GPIO_Init(GPIOB, &GPIO_InitStructure);
	
	GPIO_WriteBit(GPIOB, GPIO_Pin_12, Bit_RESET);
	GPIO_WriteBit(GPIOB, GPIO_Pin_13, Bit_RESET);
	GPIO_WriteBit(GPIOB, GPIO_Pin_15, Bit_RESET);
}

作用:使能相应管脚,并将其全部拉低。

2.void HC595_Send_Byte(unsigned char byte)

/*********************************
参数:byte 所传入的数据,
  			是要显示的数字或者是对应显示的位数
**********************************/
/************************
VCC------------>供电
DIO------------>PB15
RCLK------------>PB12  
SCLK------------>PB13  
GND------------>接地
**************************/
//串入
void HC595_Send_Byte(unsigned char byte)
{
	unsigned int i;
	
	for(i = 0; i < 8; i++)
	{
		if(byte & 0x80)
			GPIO_WriteBit(GPIOB, GPIO_Pin_15, Bit_SET);
		else
			GPIO_WriteBit(GPIOB, GPIO_Pin_15, Bit_RESET);
		
		GPIO_WriteBit(GPIOB, GPIO_Pin_13, Bit_RESET);
		Delay_us(10);
		GPIO_WriteBit(GPIOB, GPIO_Pin_13, Bit_SET);
		Delay_us(10);
		
		byte <<= 1;
	}
}

作用:发送字节到74HC595芯片的缓冲器中。
分析:发送的过程很重要,需要搞清所使用的芯片是先低位发送还是先高位。
如图:
时序图
由74HC595的芯片手册可以知道:74HC595芯片的发送顺序是由QA,一直到QH,到目前为止,我们虽然清楚了该芯片的发送顺序,但是仍然不知道QA和QH究竟哪个是高位。
而在该数据手册中,还可以发现:

这张图很明显的说明了问题的关键——QA是高位,QH是低位。
由此我们知道了这个芯片是先写入最高位,再依次发送到最低位。所以在该段函数中,我们通过该数据与(&)0x80,取出最高位,通过DIO口输入1/0,由于之前所提到的,唯有SCLK到达上升沿的时候,数据才能传入,于是我们在每次循环输入数据的时候,都将SCLK(PB13)拉高10us ,然后再将所要输入的byte左移一位,继续循环。
由于两片74HC595芯片的作用,当第一片74HC595芯片的缓冲区八位都存在数据的时候,此时DIO口继续输入数据,则将会将已有数据顶置到下一片74HC595的缓冲区。
因此所要显示数字的数据传入完成后,继续传入其所要求显示的数码管位置,该过程也同理于数字数据的传输过程。

3.void HC595_Send_Data(unsigned char num, unsigned char show_bit);

/*********************************
参数:num 所要显示的数字,
	show_bit 所显示的位数
**********************************/
/************************
VCC------------>供电
DIO------------>PB13
RCLK------------>PB12  
SCLK------------>PB15  
GND------------>接地
**************************/
//并出
void HC595_Send_Data(unsigned char num, unsigned char show_bit)
{
	HC595_Send_Byte(num);
	HC595_Send_Byte(1 << show_bit);  
	
	GPIO_WriteBit(GPIOB, GPIO_Pin_12, Bit_RESET);
	Delay_us(10);
	GPIO_WriteBit(GPIOB, GPIO_Pin_12, Bit_SET);
	Delay_us(10);
}


作用:控制4位数码管的显示位置以及显示的数字
分析:该函数使用之前所定义和编写的void HC595_Send_Byte(unsigned char byte)函数继续工作,需要注意的是,当数据和位置全部输入后,需要拉高RCLK(PB12)的电平,使其完成显示。

4.void display(unsigned int n)

最为关键的显示函数


void display(unsigned int n)
{
	static unsigned int thousand_bit, hundred_bit, ten_bit, single_bit;
	
	thousand_bit = n / 1000;
	hundred_bit = (n % 1000) / 100;
	ten_bit = n % 1000 % 100 / 10;
	single_bit = n % 10;
	
	HC595_Send_Data(num[thousand_bit], 3);
	HC595_Send_Data(num[hundred_bit], 2);
	HC595_Send_Data(num[ten_bit], 1);
	HC595_Send_Data(num[single_bit], 0);
}

由于void HC595_Send_Data(unsigned char num, unsigned char show_bit)此函数的显示问题,一次只能显示一个位置的数字,因此需要搭配主函数中的while(1){ }继续循环显示,利用人眼捕获的时间差,从而动态显示全部的数字。

总结

以上便是4位数码管的显示方法以及其分析了。
如有不足,还希望指出。

以下是使用74HC595驱动四位数码管STM32F103库代码: ```c #include "stm32f10x.h" #define DATA_PIN GPIO_Pin_7 #define LATCH_PIN GPIO_Pin_6 #define CLOCK_PIN GPIO_Pin_5 void HC595_Init(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); GPIO_InitTypeDef GPIO_InitStructure; GPIO_InitStructure.GPIO_Pin = DATA_PIN | LATCH_PIN | CLOCK_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure); } void HC595_WriteByte(unsigned char data) { GPIO_ResetBits(GPIOB, LATCH_PIN); for (int i = 0; i < 8; i++) { if (data & 0x80) GPIO_SetBits(GPIOB, DATA_PIN); else GPIO_ResetBits(GPIOB, DATA_PIN); GPIO_SetBits(GPIOB, CLOCK_PIN); GPIO_ResetBits(GPIOB, CLOCK_PIN); data <<= 1; } GPIO_SetBits(GPIOB, LATCH_PIN); } void HC595_DisplayDigit(unsigned char digit, unsigned char position) { static const unsigned char digitCode[] = { 0x3F, // 0 0x06, // 1 0x5B, // 2 0x4F, // 3 0x66, // 4 0x6D, // 5 0x7D, // 6 0x07, // 7 0x7F, // 8 0x6F, // 9 0x77, // A 0x7C, // b 0x39, // C 0x5E, // d 0x79, // E 0x71 // F }; unsigned char code = digit < sizeof(digitCode) ? digitCode[digit] : 0; code |= position << 6; HC595_WriteByte(code); } void HC595_DisplayNumber(unsigned int number) { unsigned char digit[4]; digit[3] = number % 10; digit[2] = (number / 10) % 10; digit[1] = (number / 100) % 10; digit[0] = (number / 1000) % 10; HC595_DisplayDigit(digit[0], 0); HC595_DisplayDigit(digit[1], 1); HC595_DisplayDigit(digit[2], 2); HC595_DisplayDigit(digit[3], 3); } ``` 使用方法: ```c int main(void) { HC595_Init(); while (1) { for (int i = 0; i < 10000; i++) { HC595_DisplayNumber(i); for (int j = 0; j < 100000; j++); } } } ``` 其中,`HC595_DisplayDigit`函数用于在指定位置显示一个数字,`HC595_DisplayNumber`函数用于在四个位置上分别显示一个数字。在`main`函数中,可以使用`HC595_DisplayNumber`函数不断循环显示数字。
评论 29
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值