51单片机74HC595驱动LCD1602扩展IO口

51单片机74HC595驱动LCD1602扩展IO口

一般我们用51驱动LCD1602的时候会选择P0口连接LCD1602的D0~D7,和另外三个IO控制RS,RW和EN。
只是驱动一个1602就占用了11个IO口。 而如果我们把1602的D0~D7用一个8位并行输出的芯片来传输数据不就节省了一部分IO口吗?

74HC595
74HC595是一个8位串行输入、并行输出的位移缓存器。
proteus仿真使用图如下
在这里插入图片描述
74HC595与单片机相接只需三条线。Q0~Q7与1602的 D0–D7相接。
DS是74HC595的8位数据串行输入端。SH_CP的8位串行数据输入的时钟线,在上升沿时把数据送到74HC595。当把数据传输完时,给ST_CP一个上升沿数据就从Q0–Q7并行输出。

74HC595的51驱动程序

/* 74HC595的51驱动程序 */
#include "reg52.h"
#include <intrins.h>

sbit SH_CP = P2^0;
sbit DS    = P2^1;
sbit ST_CP = P2^2;

//74HC595驱动程序
void HC595_Data(u8 sum)
 {
	 u8 k;
	 _nop_();
	 ST_CP = 0;
	 for(k=0;k<8;k++)
	 {
		SH_CP = 0;       //先把SH_CP引脚拉低
		DS = (sum&0x80); //取高位
		SH_CP = 1; 
		sum<<= 1;        //数据左移1位,把sum由高到低一位一位传输给74HC595
	 }
	 ST_CP = 1;         //给ST_CP一个上升沿,把数据并行输出
 }
 
void main()
{
 HC595_Data(0x15); //让74HC595输出 0001 0101
 while(1)
	{
	}
}

在这里插入图片描述
单独驱动74HC595的程序已经写好。那接下来就是驱动LCD1602了。

proteus仿真电路图如下
在这里插入图片描述
接下来废话不多说直接上程序

#include "reg52.h"
#include <intrins.h>

sbit RS = P1^0;
sbit RW = P1^1;
sbit E = P1^2;

sbit SH_CP = P2^0;
sbit DS    = P2^1;
sbit ST_CP = P2^2;

void LCD1602_Write_DAT(u8 dat);
void LCD1602_Write_CMD(u8 cmd);
void LCD1602_Display_Char(bit x,u8 y,u8 *s);
void LCD1602_Position(bit x,u8 y);
void LCD1602_Display_Data(bit x,u8 y,u8 dat);
void HC595_Data(u8 sum);

u16 num;
u16 i;
void main()
{
	num = 2021;
	i   = 1120 ;
	
	LCD1602_Write_CMD(0x06);	//指针自增
	LCD1602_Write_CMD(0x0C);	//显示开,光标关
	LCD1602_Write_CMD(0x38);	//8位接口,两行显示
  	LCD1602_Write_CMD(0x01);	//清屏

/* 关于这个LCD1602_Write_DAT('0'); 语句。我给大家解释一下。  
可能大家看完这个会有疑问。为什么要给1602发送一个0呢?而且也不指定位置呢?
这是因为不知道是我程序的问题还是电路的问题(严重怀疑是74HC595驱动程序的问题)
经过多次的调试发现给1602发送的第一个数据1602接收不到(给1602的第一个命令可以接收到)
第二次及以后的数据可以正常接收。
所以在显示字符串语句的前面,要有一个发送数据的函数。不然字符串的第一个字母则会显示不正常
本人水平有限请见谅
 */
	LCD1602_Write_DAT('0');
		
	LCD1602_Display_Char(0,6,"TYQ");
	while(1)
	{
		//显示2021年
	   LCD1602_Display_Data(1,0,num/1000+'0'); //       显示千位 2
	   LCD1602_Display_Data(1,1,num%1000/100+'0'); //   显示百位 0
	   LCD1602_Display_Data(1,2,num%100/10+'0'); //     显示十位 2
	   LCD1602_Display_Data(1,3,num%10+'0'); //         显示个位 1
		LCD1602_Display_Char(1,4,"year");
		//显示11 月
	   LCD1602_Display_Data(1,8,i/1000+'0'); //       显示千位 1
	   LCD1602_Display_Data(1,9,i%1000/100+'0'); //   显示百位 1
	   LCD1602_Display_Char(1,10,"moon");							
       //显示20		
	   LCD1602_Display_Data(1,14,i%100/10+'0'); //     显示十位 2
	   LCD1602_Display_Data(1,15,i%10+'0'); //         显示个位 0
		
	}
}

//LCD1602 位置显示指函数
void LCD1602_Position(bit x,u8 y)
{
    u8 position;
	if(x==0)
		position = 0x80 + y;
	else
		position = 0xC0 + y;
	LCD1602_Write_CMD(position);
}

//LCD1602写数据函数
void LCD1602_Write_DAT(u8 dat)
{
	_nop_();
	_nop_();
	RS = 1;
	RW = 0;
	_nop_();
	HC595_Data(dat);
	_nop_();
	E = 0;
	E = 1;
	E = 0;
}

//LCD1602写命令函数
void LCD1602_Write_CMD(u8 cmd)
{
	_nop_();
	_nop_();
	RS = 0;
	RW = 0;;
	HC595_Data(cmd);
	E = 1;
	E = 0;
}

//LCD1602指定位置显示字符串函数
void LCD1602_Display_Char(bit x,u8 y,u8 *s) 
{
   LCD1602_Position(x,y);
	while(*s)
		LCD1602_Write_DAT(*s++);
}

//LCD1602指定位置显示数字函数	
void LCD1602_Display_Data(bit x,u8 y,u8 dat)  
{
	LCD1602_Position( x, y);
	LCD1602_Write_DAT(dat);
}

//74HC595驱动函数
void HC595_Data(u8 sum)
 {
	 u8 k;
	 ST_CP = 0;
	 for(k=0;k<8;k++)
	 {
		SH_CP = 0;
		DS = (sum&0x80);
		SH_CP = 1; 
		sum<<= 1;
	 }
	 ST_CP = 1;
 }

在这里插入图片描述
相关程序和仿真电路存放至网盘
链接:https://pan.baidu.com/s/1yWkaex99k1dm1rkal-8E4w
提取码:TYQT
联系方式:2187066701@qq.com

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值