51单片机:IO扩展 74HC595


前言

由于C51芯片IO口有限,当需要控制更多外围设备时,需要对IO口进行扩展。通过74HC595芯片可以通过使用3个IO口,扩展出8个IO口(串转并)。

通过74HC595控制8*8的LED点阵实验,学习IO扩展。

一、74HC595及点阵原理

在这里插入图片描述

引脚描述
10低电平复位引脚(给低电平即清除数据复位)
13低电平输出有效(给低电平才有输出)
14串行数据输入
11移位寄存器时钟输入
12存储寄存器时钟输入
9串行数据输出
1-7,15并行数据输出
  1. 10引脚在51单片机中始终接高电平,即在51单片机中无法对74HC595复位 ,在实际编写程序中不用管该引脚。
  2. 13引脚OE给低电平才能有输出,即原理图最左边将OE与GND连接。在51单片机实物中,将最右边数码管旁边的标号为J24的短接片插到左边。此时74HC595才能产生有效输出。
  3. 在14引脚输入数据时,不能直接将十六进制数据赋值给P34。需要将数据一位一位的输入。以0x80为例,转换为二进制为1000 0000。需要从高位至低位依次获取每一位数据,将每一位数据赋值给P34。
  4. 在P34获取到一位数据后,给11引脚一个从低电平到高电平的上升沿,将数据写入移位寄存器。
  5. 当将所有数据写入移位寄存器后,给12引脚一个从低电平到高电平的上升沿,将数据写入存储寄存器。此时在1-7,15引脚并行输出中才能获得数据。

在这里插入图片描述
在这里插入图片描述
6. 并行输出中15引脚为最低位,7引脚为最高位

二、点阵代码

1.时延文件

delay.h

#ifndef _DELAY_H__
#define _DELAY_H__
typedef unsigned int uint;
typedef unsigned char uchar;

/*10us=1,时延为10us*/
void delay_10us(uint _10us){
	while(_10us--){}
}

/*1ms=1,时延为1ms*/
void delay_1ms(uint _1ms){
	 _1ms*=110;
	 while(_1ms--){
	 }
}

#endif

2.74HC595控制代码

Hc595_utils.h

#ifndef _HC595_UTILS_H
#define _HC595_UTILS_H
#include "reg52.h"
#include "delay.h"

sbit SHCP = P3^6; //移位寄存器时钟
sbit STCP = P3^5; //存储寄存器时钟
sbit SER = P3^4; //串行数据输入

/*写入数据到HC595芯片*/
void Write_2_Hc595(uchar dat){
	uint i;
	for(i=0;i<8;++i){
		/*获取高位*/
		SER = dat>>7;
		/*获得除高位后的数据*/
		dat<<=1;
		/*产生SHCP电压上升沿*/
		SHCP=0;
		delay_10us(1);//时延10us
		SHCP=1;//电压上升沿,写入移位寄存器		
	}
	/*写入存储寄存器*/
	STCP=0;
	delay_10us(1);
	STCP=1;//STCP上升沿,写入存储寄存器
}

/*显示8*8点阵的一张图像*/
void show_pic(uchar *pic){//传入一张图像的数组
	uint j=100;
	while(j--){//循环若干次,留足视觉暂存
		uint i;
		for(i=0 ; i<8;++i){//一张图有8个数据,
			P0=~(0x80>>i);
			/*写入数据到HC595*/
			Write_2_Hc595(pic[i]);			
			/*位清零,消残影*/
			delay_10us(1);
			Write_2_Hc595(0x00);				
		}
	}
}
#endif

3. 主函数

#include "Hc595_utils.h"

void main(){
	uchar pic[10][8]={{0x00,0x00,0x7E,0x10,0x10,0x1E,0x00,0x00},//h
					{0x00,0x00,0x7C,0x54,0x54,0x74,0x00,0x00},//e
					{0x00,0x00,0x79,0x09,0x09,0x7F,0x00,0x00},//y
					
					{0x00,0x00,0x7E,0x50,0x50,0x50,0x00,0x00},//f
					{0x00,0x3C,0x02,0x02,0x02,0x3C,0x00,0x00},//u
					{0x00,0x3C,0x42,0x42,0x42,0x42,0x00,0x00},//c
					{0x00,0x00,0x7E,0x18,0x24,0x42,0x00,0x00}, //k
					
					{0x00,0x00,0x79,0x09,0x09,0x7F,0x00,0x00},//y
					{0x00,0x3C,0x42,0x42,0x42,0x42,0x3C,0x00},//o
					{0x00,0x3C,0x02,0x02,0x02,0x3C,0x00,0x00}//u
					};
	
	while(1){
		uint i;
		for(i=0 ; i<10;++i){
			delay_1ms(100);
		  	show_pic(&(pic[i]));
		  	delay_1ms(100);
		}
		
	}
}

代码解释

  1. 数据类型须为unsigned char类型?
    因为如果为unsigned int类型无法正常显示。
  2. P0=~(0x80>>i)
    74HC595控制8*8LED点阵的行,P0控制点阵的列。以pic[10][8]中的任意一行为例,一行中的数据有8个元素,第一个元素表示选中第一列进行显示,第一列的如何显示由第一个元素的值即595的输出控制。以此类推。
    i=1为控制第二列显示,0x80转换为二进制为1000 0000 , 0x80>>1表示右移1位,即为0100 0000。取反码为1011 1111。赋值给P0即表示第二列为低电平,由LED管的方向可知,为低电平时有效,即控制第二列进行显示。至于如何显示,则由Write_2_Hc595(pic[i]);中的pic[i]控制。

三、如何获取点阵的行显示数据?

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

总结

如何控制点阵不重要,如何学会看芯片文档,如何进行IO扩展很重要

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值