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