您的当前位置:首页用74hc595把lcd1602改成串行数据接口

用74hc595把lcd1602改成串行数据接口

2021-08-21 来源:乌哈旅游
 用 74HC595 把 LCD1602 改成串行接口形式的案例

LCD1602 的接口形式是并行的,它有 8 条数据线、3 条控制线。这样就需要 11 条线来控制它的正常工作。虽然它还可以工作在 4 位数据线的形式,最精简的形式是 6 条线。

有位网友想要使用 74HC595 进行串-并转换,想要用 4 条线来控制 LCD1602。可见链接: 。

多用了一块芯片,省下了单片机的引脚,这也算是一种方法吧,在系统规模较大、资源紧张的条件下,还是值得应用的。

74HC595 是“串入并出”的移位寄存器芯片,它需要用 3 条线控制数据的输入,才能正常的输出 8 位数据。

有了 8 位数据,这时,LCD1602 还需要至少两条控制线。

经过做而论道的精心设计,分时使用这些控制线,最终,仅仅使用了 3 条线,就完成了对 74HC595 和 LCD1602 的有效控制 !

这要比前面的网友提出的 4 条线的方案,还要更加精简,节省率提高了 25%。

精简后的电路图如下:

图中是一个数字钟,是为另一个网友编写的,可以参考下面的链接:

由于 LCD1602 的驱动电路发生了改变,所以数字钟的程序,也要相应的修改。那么,针对本电路的程序如下:

//===================================================

#include<>

#define uchar unsigned char #define uint unsigned int

#define KEY_IO P3

sbit LCD_RS = P2^0; sbit LCD_EN = P2^2;

sbit SCK = P2^0; sbit SDI = P2^1; sbit RCK = P2^2;

sbit SPK = P1^2; sbit LED = P2^4;

bit new_s, modify = 0;

char t0, sec = 50, min = 59, hour = 23;

char code LCD_line1[] = \"Designed by ZELD\"; char code LCD_line2[] = \"Timer: 00:00:00 \"; char Timer_buf[] = \"23:59:50\";

//--------------------------------------------------- void delay(uint z) {

uint x, y;

for(x = z; x > 0; x--) for(y = 100; y > 0; y--); }

//---------------------------------------------------

void write_595(uchar date)//写入595 {

uchar i;

for(i = 0; i < 8; i++) { SCK = 0; SDI = date & 0x80; SCK = 1; date <<= 1; } }

//--------------------------------------------------- void W_LCD_Com(uchar com) //写指令 {

write_595(com); LCD_RS = 0;//写入指令

RCK = 1; RCK = 0; //令595输出, 并用EN输出一个高脉冲 }

//--------------------------------------------------- void W_LCD_Dat(uchar dat) //写数据

{

write_595(dat); LCD_RS = 1;//写入数据

RCK = 1; RCK = 0; //令595输出, 并用EN输出一个高脉冲 }

//--------------------------------------------------- void W_LCD_STR(uchar *s) //写字符串 {

while(*s) W_LCD_Dat(*s++); }

//--------------------------------------------------- void W_BUFF(void) //填写显示缓冲区 {

Timer_buf[7] = sec % 10 + 48; Timer_buf[6] = sec / 10 + 48; Timer_buf[4] = min % 10 + 48; Timer_buf[3] = min / 10 + 48; Timer_buf[1] = hour % 10 + 48;Timer_buf[0] = hour / 10 + 48; W_LCD_Com(0xc0 + 7); W_LCD_STR(Timer_buf);

}

//--------------------------------------------------- uchar read_key(void) {

uchar x1, x2; KEY_IO = 255; x1 = KEY_IO; if (x1 != 255) { delay(100); x2 = KEY_IO;

if (x1 != x2) return 255; while(x2 != 255) x2 = KEY_IO; if (x1 == 0x7f) return 0; else if (x1 == 0xbf) return 1; else if (x1 == 0xdf) return 2; else if (x1 == 0xef) return 3;

else if (x1 == 0xf7) return 4; }

return 255; }

//--------------------------------------------------- void Init() {

W_LCD_Com(0x38); delay(50); W_LCD_Com(0x38); delay(50); W_LCD_Com(0x0c); W_LCD_Com(0x06);

W_LCD_Com(0x01); delay(50);

W_LCD_Com(0x80); W_LCD_STR(LCD_line1); W_LCD_Com(0xC0); W_LCD_STR(LCD_line2);

TMOD = 0x01; //T0定时方式1

TH0 = 0x4c;

TR0 = 1; //启动T0

PT0 = 1; //高优先级, 以保证定时精度 ET0 = 1; EA = 1; }

//--------------------------------------------------- void main() {

uint i, j; uchar Key; Init();

while(1) {

//-------------------------------

if (new_s) { //如果出现了新的一秒, 修改时间

new_s = 0; sec++; sec %= 60; if(!sec) { min++; min %= 60; if(!min) { hour++; hour %= 24;} }

W_BUFF(); //写显示 //------------------------------- if (!sec && !min) { //整点报时 for (i = 0; i < 200; i++) {

SPK = 0; for (j = 0; j < 100; j++); SPK = 1; for (j = 0; j < 100; j++); } } }

//------------------------------- Key = read_key(); //读出按键 switch(Key) { //分别处理四个按键 case 0: modify = 1; break;

case 1: if(modify) {min++; min %= 60; W_BUFF(); break;} case 2: if(modify) {hour++; hour %= 24; W_BUFF(); break;} case 3: modify = 0; break; } } }

//--------------------------------------------------- void timer0(void) interrupt 1 //T0中断函数, 50ms执行一次 {

TH0 = 0x4c;

t0++; t0 %= 20; //20, 一秒钟 if(t0 == 0) {new_s = 1; LED = ~LED;} if(modify) LED = 0; }

//===================================================

欢迎到“做而论道”的百度空间访问,这里有许多新奇实用的电路和程序。

因篇幅问题不能全部显示,请点此查看更多更全内容