HC595级联原理及实例 - STM32

        74HC595的最重要的功能就是:串行输入,并行输出。其次,74HC595里面有2个8位寄存器:移位寄存器、存储寄存器。74HC595的数据来源只有一个口,一次只能输入一个位,那么连续输入8次,就可以积攒为一个字节了。

引脚图

14脚:DIN(SER),串行数据输入引脚

13脚:OE,  输出使能控制脚,它是低电才使能输出,所以接GND

12脚:RCK,存储寄存器时钟输入引脚。上升沿时,数据从移位寄存器转存带存储寄存器。

11脚:SCK,移位寄存器时钟引脚,上升沿时,移位寄存器中的bit 数据整体后移,并接受新的bit(从SER输入)。

10脚:SCLR,低电平时,清空移位寄存器中已有的bit数据,一般不用,接高电平即可。

9 脚 :串行数据出口引脚。当移位寄存器中的数据多于8bit时,会把已有的bit“挤出去”,就是从这里出去的。用于595的级联。

Qx:并行输出引脚

使用参数

VCC:2V~6V,5V最好

I Qn:+- 35mA

移位寄存器

74HC595的14脚:DIN,是串行数据输入口。595的数据来源只有这一个口,一次只能输入一个位,那么连续输入8次,就可以积攒为一个字节了。

74HC595的11脚,(shift register clock input) 移位寄存器时钟引脚。上升沿有效。
首先我们要介绍这个引脚的作用,当一个新的位数据要进来时,已经进入的位数据就在移位寄存器时钟脉冲的控制下,整体后移,让出位置。

上升沿:电平从低到高的那个过程。移位寄存器时钟在上升沿这个过程中才起作用。

存储寄存器

595是怎么将移位寄存器的数据转移到存储寄存器。存储寄存器是直接和8个输出引脚相通的,将移位寄存器的数据转移到存储寄存器后,Q0 Q1 Q2 Q3 Q4 Q5 Q6 Q7 就可以接收到我们开始输入的一个字节的数据。所谓存储寄存器,就是数据可以存在这个寄存器中,并不会随着一次输出就消失,只要595不断电,也没有新的数据从移位寄存器中过来,数据就一直不变且有效。新的数据过来后,存储寄存器中的数据就会被覆盖更新。

74HC595的12脚: (storage register clock input ) 存储寄存器时钟
数据从位移寄存器转移到存储寄存器,也是需要时钟脉冲驱动的,这就是12脚的作用。它也是上升沿有效。

74HC595级联

通过上面的介绍,见识到595的厉害了吧。138译码器通过3个输入口控制8个输出口,而且还只能是特定的8个输出值,而595只用了一个输入口就可以输任意的8位数据,可谓短小精悍。

你觉的1位控制8位输出还不够?

在上面的程序中用到的9脚,没用起作用,如果要让2个595串联起来的话,就需要它了。想一下,我们将移位寄存器的8个位填满后,再往移位寄存器中塞一个会怎么样?也许你想到了。对!移位寄存器的最后一个位数据会被挤出去,从哪里出去?就是从9脚输出的。如果我们把第一个595的9脚连接到第二个的串行数据输入脚SER,那么,就形成了595的级联。这样,如果我们用2个595组合成了一个新的超级595,这个超级595的移位寄存器和存储寄存器的容量都翻倍了,1口控制16口,有木有!你还可以继续级联下去!

74HC595级联实例

#ifndef __HC_595_H__
#define __HC_595_H__	

#include "stm32h7xx_hal.h"


// HC595的16位输出端定义位号
// Q0 - 0 .....Q15 -15
#define DIO_HC_A0           (6)
#define DIO_HC_B0           (7)
#define DIO_HC_C0           (0)
#define DIO_HC_A1           (15)
#define DIO_HC_B1           (8)
#define DIO_HC_C1           (1)
#define DIO_HC_A2           (10)
#define DIO_HC_B2           (13)
#define DIO_HC_C2           (14)
/* RLY */
#define DIO_RLY0            (5)
#define DIO_RLY1            (2)
/* S1 S2 */
#define DIO_S1              (11)
#define DIO_S2              (12)
/* CUR */
#define DIO_CUR             (9)

/*----------------------------------------- HC595 引脚配置宏 -----------------------------------------------*/

#define HC_595_DIN_ENABLE     		 __HAL_RCC_GPIOE_CLK_ENABLE()			// 使能DIN引脚时钟
#define HC_595_DIN_PORT   			 GPIOE                 					// DIN引脚端口
#define HC_595_DIN_PIN     		 	 GPIO_PIN_1  							// DIN引脚
         
#define HC_595_RCK_ENABLE     		 __HAL_RCC_GPIOE_CLK_ENABLE()			// 使能RCK引脚时钟
#define HC_595_RCK_PORT   			 GPIOE                 					// RCK引脚端口
#define HC_595_RCK_PIN     		 	 GPIO_PIN_3  							// RCK引脚

#define HC_595_SCK_ENABLE     		 __HAL_RCC_GPIOE_CLK_ENABLE()			// 使能SCK引脚时钟
#define HC_595_SCK_PORT   			 GPIOE                 					// SCK引脚端口
#define HC_595_SCK_PIN     		 	 GPIO_PIN_2  							// SCK引脚

#define HC_595_SCLR_ENABLE     		 __HAL_RCC_GPIOC_CLK_ENABLE()			// 使能SCLR引脚时钟
#define HC_595_SCLR_PORT   			 GPIOC                 					// SCLR引脚端口
#define HC_595_SCLR_PIN     		 GPIO_PIN_4  								// SCLR引脚

/*-------------------------------------------- IO口操作 ---------------------------------------------------*/   

#define HC_595_DIN(a)	if (a)	\
										HAL_GPIO_WritePin(HC_595_DIN_PORT, HC_595_DIN_PIN, GPIO_PIN_SET); \
									else		\
										HAL_GPIO_WritePin(HC_595_DIN_PORT, HC_595_DIN_PIN, GPIO_PIN_RESET)	

#define HC_595_RCK(a)	if (a)	\
										HAL_GPIO_WritePin(HC_595_RCK_PORT, HC_595_RCK_PIN, GPIO_PIN_SET); \
									else		\
										HAL_GPIO_WritePin(HC_595_RCK_PORT, HC_595_RCK_PIN, GPIO_PIN_RESET)	
									
#define HC_595_SCK(a)	if (a)	\
										HAL_GPIO_WritePin(HC_595_SCK_PORT, HC_595_SCK_PIN, GPIO_PIN_SET); \
									else		\
										HAL_GPIO_WritePin(HC_595_SCK_PORT, HC_595_SCK_PIN, GPIO_PIN_RESET)	

#define HC_595_SCLR(a)	if (a)	\
										HAL_GPIO_WritePin(HC_595_SCLR_PORT, HC_595_SCLR_PIN, GPIO_PIN_SET); \
									else		\
										HAL_GPIO_WritePin(HC_595_SCLR_PORT, HC_595_SCLR_PIN, GPIO_PIN_RESET)	
																		

// 函数声明
void HC_595_GPIO_Config(void);
void HC_595_Send_Byte(unsigned short Q15_Q0);
void HC595_Write_QX(unsigned short index,unsigned short sta);
									
#endif
#include "hc595.h"

static unsigned short Q0_Q15_S = 0;

/*****************************************************************************************
*	函数名: HC595_Delay
*	入口参数: t - 延时时间,以时钟周期数为单位
*	返回值: 无
*	函数功能: 简单延时函数
*	说明: 为了移植的简便性且对延时精度要求不高,所以不需要使用定时器做延时
******************************************************************************************/
void HC595_Delay(unsigned int t)
{
	while(t--); // 简单的循环延时,延时时间由入口参数 t 决定
}


/*****************************************************************************************
*	函数名: HC_595_GPIO_Config
*	入口参数: 无
*	返回值: 无
*	函数功能: 初始化移位寄存器的 GPIO 口
*	说明: 配置数据输入引脚(DIN)、存储寄存器时钟引脚(RCK)、移位寄存器时钟引脚(SCK)和清除引脚(SCLR)为推挽输出模式,
*		 不带上下拉,速度配置为低速。然后将这些引脚初始化,并设置初始电平状态。
******************************************************************************************/
void HC_595_GPIO_Config(void)
{	
	GPIO_InitTypeDef GPIO_InitStruct = {0};
	
	// 初始化IO口时钟
	HC_595_DIN_ENABLE;
	HC_595_RCK_ENABLE;
	HC_595_SCK_ENABLE;	
	HC_595_SCLR_ENABLE;	
	
	// 配置DIN引脚
	GPIO_InitStruct.Pin 			= HC_595_DIN_PIN;
	GPIO_InitStruct.Mode 			= GPIO_MODE_OUTPUT_PP;	// 推挽输出
	GPIO_InitStruct.Pull 			= GPIO_NOPULL;			// 不带上下拉
	GPIO_InitStruct.Speed 			= GPIO_SPEED_FREQ_LOW;	// 低速
	HAL_GPIO_Init(HC_595_DIN_PORT, &GPIO_InitStruct);

	// 配置RCK引脚
	GPIO_InitStruct.Pin 			= HC_595_RCK_PIN;
	HAL_GPIO_Init(HC_595_RCK_PORT, &GPIO_InitStruct);		

	// 配置SCK引脚
	GPIO_InitStruct.Pin			= HC_595_SCK_PIN;
	HAL_GPIO_Init(HC_595_SCK_PORT, &GPIO_InitStruct);				

	// 配置SCLR引脚
	GPIO_InitStruct.Pin 			= HC_595_SCLR_PIN;
	GPIO_InitStruct.Pull 			= GPIO_PULLUP;			// 上拉
	HAL_GPIO_Init(HC_595_SCLR_PORT, &GPIO_InitStruct);

	// 初始化引脚状态
	HAL_GPIO_WritePin(HC_595_RCK_PORT, HC_595_RCK_PIN, GPIO_PIN_RESET);	// RCK输出低电平
	HAL_GPIO_WritePin(HC_595_SCK_PORT, HC_595_SCK_PIN, GPIO_PIN_RESET);	// SCK输出低电平
	HAL_GPIO_WritePin(HC_595_SCLR_PORT, HC_595_SCLR_PIN, GPIO_PIN_SET);	// SCLR输出高电平
}


/*****************************************************************************************
*	函数名: HC_595_Send_Byte
*	入口参数: Q15_Q0 - 要发送的16位数据
*	返回值: 无
*	函数功能: 向移位寄存器发送一个16位数据
*	说明: 首先设置移位寄存器的控制引脚为初始状态,然后逐位发送数据,最后将数据加载到移位寄存器中。
******************************************************************************************/
void HC_595_Send_Byte(unsigned short Q15_Q0)
{
	// 设置移位寄存器的控制引脚初始状态
	HC_595_DIN(0);
	HC_595_RCK(0);
	HC_595_SCK(0);

	HC595_Delay(10);	
	
	for( int i = 0 ; i < 16 ; i ++ )
	{
		// 逐位发送数据
		HC_595_SCK(0);
		if( Q15_Q0 & 0x8000 )
		{
			HC_595_DIN(1);
		}
		else
		{
			HC_595_DIN(0);
		}
		HC595_Delay(1);		
		HC_595_SCK(1);		// 移位寄存器输入一个字节数据
		HC595_Delay(1);
		Q15_Q0 <<= 1;
	}
	
	// 将数据加载到移位寄存器中
	HC_595_RCK(1);			// 存储寄存器输入,Q1-Q15输出
	HC595_Delay(1);
	
	HC_595_RCK(0);
	HC_595_SCK(0);
}

/* w_gpio_sem */
static unsigned char w_gpio_sem = 0;

/*****************************************************************************************
*	函数名: HC595_Write_QX
*	入口参数: index - 输出引脚的编号,sta - 输出状态,1为高电平,0为低电平
*	返回值: 无
*	函数功能: 控制移位寄存器的输出引脚状态
*	说明: 根据输入的输出引脚编号和状态,更新移位寄存器中对应引脚的状态,并发送更新后的状态到移位寄存器。
******************************************************************************************/
void HC595_Write_QX(unsigned short index,unsigned short sta)
{
	// 检查是否有其他操作在进行
	if( w_gpio_sem )
	{
		return;
	}
	
	// 锁定
	w_gpio_sem = 1;
	
	// 检查输出引脚编号是否合法
	if( index > 16 )
	{
		return;
	}
	
	// 根据状态设置输出引脚状态
	if( sta )
	{
		Q0_Q15_S |= ( 1 << index );
	}
	else
	{
		Q0_Q15_S &=~ ( 1 << index );
	}
	
	// 更新移位寄存器中的输出状态
	HC_595_Send_Byte(Q0_Q15_S);
	
	// 解锁
	w_gpio_sem = 0;	
}

  1. HC_595_GPIO_Config: 初始化HC595芯片的GPIO口,包括数据输入引脚(DIN)、存储寄存器时钟引脚(RCK)、移位寄存器时钟引脚(SCK)和清除引脚(SCLR)。配置这些引脚为推挽输出模式,并设置初始电平状态。

  2. HC_595_Send_Byte: 向HC595芯片发送一个16位数据,通过移位寄存器将数据加载到芯片中,控制输出引脚的状态。

  3. HC595_Write_QX: 根据输入的输出引脚编号和状态,更新移位寄存器中对应引脚的状态,并通过 HC_595_Send_Byte 函数发送更新后的状态到移位寄存器。

  4. HC595_Delay: 简单的延时函数,用于产生一定时间的延时,以时钟周期数为单位。

注意 

HC595的DIN输入端输出的数据会从Q0开始,继续输入,之前的Q0数据会移位至Q1,一直循环。

完整的输入一个字节数据,假设输入的是unsigned char Q_byte =(高)1011 1111(低) 。因为HC_595_Send_Byte程序中是从高位开始发送的,最高位的数据再经过完整的一个字节循环后就被push到了Q7(很像队列)

Q0 = 1;Q1=1;Q2=1;Q3=1;Q4=1;Q5=1;Q6=0;Q7=1;

我们要改变Q0-Q7中的某个位做法:

假设改变Q3位,则把 (高)1011 0111(低)重新写入HC595中。

Q_byte = Q_byte & ~ (1<<3)  = 1011 1111 & 1111 0111 = 1011 0111;

这是一个595的情况,级联情况下同理,代码,处理的就是级联情况。所以应用HC595_Write_QX函数就可以操作Q0-Q15任意一个输出位的电平了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/408136.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MiKTeX安装后,Latex编译后PDF无法预览,是灰色的

解决方式删掉编译器就可以&#xff0c; 即删掉MiKTeX MiKTeX安装后会将编译器默认修改为MiKTeX&#xff0c;这个时候会显示报错&#xff0c;简单粗暴的方式是删掉MiKTeX软件

python统计分析——单因素方差分析

参考资料&#xff1a;用python动手学统计学 方差分析&#xff1a;analysis of variance&#xff0c;缩写为ANOVA 1、背景知识 1.1 要使用方差分析&#xff0c;数据的总体必须服从正态分布&#xff0c;而且各个水平内部的方差必须相等。 1.2 反复检验导致显著性结果更易出现…

相机选型介绍

摄影测量中&#xff0c;相机是非常重要的角色&#xff0c;合适的相机产出合适的图像&#xff0c;得到合适的重建精度&#xff0c;这是相机的重要性。 您也许第一反应是&#xff0c;摄影测量所需的理想相机&#xff0c;是有着超高分辨率的相机&#xff0c;但事实可能并非如此&a…

太阳能光伏模型的参数确定及模型应用介绍

一、太阳能光伏模型介绍 ​ 太阳能通过光伏&#xff08;PV&#xff09;发电系统转化为电能。通过使用新材料技术&#xff0c;一直致力于提高光伏系统中太阳能电池的功率转换效率。基于钙钛矿太阳能电池的冠军器件具有24.8%的认证功率转换效率&#xff0c;仍有很大的改进空间。…

学习 LangChain 的 Passing data through

学习 LangChain 的 Passing data through 1. Passing data through2. 示例 1. Passing data through RunnablePassthrough 允许不改变或添加额外的键来传递输入。这通常与 RunnableParallel 结合使用&#xff0c;将数据分配给映射中的新键。 RunnablePassthrough() 单独调用&…

ChatGPT Plus遇到订阅被拒原因与解决方案

ChatGPT Plus被广泛认为相比普通版本更快、更强&#xff0c;并且能最先体验新功能。 很多小伙伴再订阅时遇到图片中的问题 错误提示包括这些&#xff1a; Your credit card was declined.Try paying with a debit card instead.您的信用卡被拒绝了。请尝试用借记卡支付。你的…

Data-Shape制作UI节点简介

很多小伙伴对于Data-Shape制作简单的UI都是似懂非懂&#xff0c;群里很多小伙伴也总是询问相关的问题&#xff0c;那么&#xff0c;今天我就简单举几个例子&#xff0c;专门介绍下Data-Shape创建简单UI的教程。 首先&#xff0c;本次教程&#xff0c;使用的是Data-Shape 2021.…

队列基础练习(C语言实现)

目录 队列基础练习 用队列实现栈 用栈实现队列 设计循环队列 队列基础练习 用队列实现栈 题目链接&#xff1a;225. 用队列实现栈 - 力扣&#xff08;LeetCode&#xff09; 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#xff0c;并支持普通栈的全部…

政安晨【示例演绎虚拟世界开发】(一):Cocos Creator 的 Hello World

政安晨的个人主页&#xff1a;政安晨 欢迎 &#x1f44d;点赞✍评论⭐收藏 收录专栏: AI虚拟世界大讲堂 希望政安晨的博客能够对您有所裨益&#xff0c;如有不足之处&#xff0c;欢迎在评论区提出指正。 前言 Cocos Creator是一款非常强大的游戏开发引擎&#xff0c;它有着优秀…

基于springboot+vue的安康旅游网站(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

洛谷P5723 质数口袋 题解

#题外话&#xff08;第39篇题解&#xff09;&#xff08;本题为普及-难度&#xff09; #先看题目 题目链接https://www.luogu.com.cn/problem/P5723 #思路&#xff08;看代码吧&#xff09; #代码 #include <bits/stdc.h> using namespace std; bool p(int p_i){for(i…

啊丢的刷题记录手册

1.洛谷题P1923 求第k小的数 题目描述 输入 n&#xff08;1≤n<5000000 且 n 为奇数&#xff09;个数字ai​&#xff08;1≤ai​<109&#xff09;&#xff0c;输出这些数字的第 k 小的数。最小的数是第 0 小。 请尽量不要使用 nth_element 来写本题&#xff0c;因为本题…

《优化接口设计的思路》系列:第八篇—分页接口的设计和优化

系列文章导航 第一篇—接口参数的一些弯弯绕绕 第二篇—接口用户上下文的设计与实现 第三篇—留下用户调用接口的痕迹 第四篇—接口的权限控制 第五篇—接口发生异常如何统一处理 第六篇—接口防抖(防重复提交)的一些方式 第七篇—接口限流策略 第八篇—分页接口的设计和优化 …

C语言实现简单选择排序

简单选择排序 简单选择排序的平均复杂度为 O(n2),但效率通常比相同平均复杂度的直接插入排序还要差。但由于选择排序是 内部排序&#xff0c;因此在内存严格受限的情况下还是可以用的。选择排序的原理很简单&#xff0c;如下图所示&#xff1a;持续从未处理元素中找到最小值并加…

虚拟机的四种网络模式对比

nat网络地址转换 nat网络 桥接 内网模式 仅主机

【数据结构(顺序表)】

一、什么是数据结构? 数据结构是由“数据”和“结构”两词组合而来。 什么是数据&#xff1f;常见的数值1、2、3、4.....、教务系统里保存的用户信息&#xff08;姓名、性别、年龄、学历等等&#xff09;、网页里肉眼可以看到的信息&#xff08;文字、图片、视频等等&#xff…

Google炸场!最强轻量级、开放模型Gemma发布,个人PC就能用,内部员工:强是强,但名字取得让我混乱

想参与根多多学术讨论&#xff0c;请加qq群 链接直达&#xff1a;00后编程交流qq群 如果想要聊天交友&#xff0c;可以加qq群 链接直达&#xff1a;00后聊天交友处cp 欢迎大家加入 不同于OpenAI的闭源大模型&#xff0c;科技巨头如Google和Meta正积极投入开放模型的开发&a…

Qt的QFileSystemModel与QTreeView、QTableView、QListView的组合使用

1.相关描述 QFileSystemModel与QTreeView、QTableView、QListView的组合&#xff0c;当QTreeView点击发生改变&#xff0c;QTableView和QListView也会发生变化 2.相关界面 3.相关代码 mainwindow.cpp #include "mainwindow.h" #include "ui_mainwindow.h"…

Flutter开发进阶之Package

Flutter开发进阶之Package 通常我们在Flutter开发中需要将部分功能与整体项目隔离&#xff0c;一般有两种方案Plugin和Package&#xff0c;Application是作为主体项目&#xff0c;Module是作为原生项目接入Flutter模块。 当独立模块不需要与原生项目通讯只需要Plugin就可以&a…

990-05产品经理:为什么商业价值是 IT 成功的关键

In today’s digital era, CIOs must shift(转移) their priorities from cost cutting to driving revenue(收入), and from process engineering to exploiting data if they want to achieve a set of broader business outcomes. Furthermore, understanding how to measur…