呵呵,不用我說你們也知道為什么只需要4根線:因為咱用了74hc595啊!因為74HC595最近我手里一抓一大把。但是STC的單片機IO口緊缺。就算使用4線驅動模式也要8個IO(還要算上背光PWM)。這一次我幾乎做到了他的一半。如果不用PWM的話。大概只需要3個IO,如果需要PWM就四個羅。
MCU當然是老掉牙的死特慘89C52羅~
也可以在595后面級聯類似三極管開關……繼電器之類的玩意。當然友情提醒,別超過4個595.否則速度會慢的可怕。我現在1個595.寫1602都可以丟掉延時函數了。
這一次硬件設計的比較簡單,但是IO全反了。所以需要占用額外的CPU時間修正,但是畢竟方便洞洞板嘛。還有大家別看到我給A和B賦了好幾次值而去“優化”代碼。如果您“優化”的話,那么就根本驅動不起來,因為51的加法減法乘除都要經過ACC和B,如果不去重新賦值的話,那么就會發生改變的現象。不過我這種位尋址軟件修正可比那種xx=xx & xx的效率高多了。如果我這么寫的話。恐怕刷一個屏幕沒1秒下不來啊。12T的51就這個速度……
不過這一次焊板子實在是太倉促了。對比度電位器焊在液晶下面,結果初始化成功了顯示兩排方塊。暈了半天。調試半天還是沒用。后來捅了一下背面就好了……暈……!
上代碼:
#include <stc89c5x.h>//STC89C52
//串行驅動1602,powered by 595
#define LCD1602_BACKLIGHT P1_3//背光
#define LCD1602_SDA P1_0//數據輸入
#define LCD1602_SCK P1_1//移位時鐘,SHCP,11pin
#define LCD1602_SCL P1_2//鎖存時鐘,STCP,12pin
/*
* 硬件連接:
* 74HC595 Q7 - 1602 RS
* GND - 1602 RW(595沒有輸入功能,所以判定忙只能*延時,另外就是MCU的速度比較的慢。所以也可以直接去掉延時。但是在STC12系列上的表現,相當的不錯
* 在傳統51上面。PWM占空比低于50%將會出現掃描線。12系列的,很高興的告訴大家。T0X12開了以后。不會有掃描線出現。但是如果一定要在傳統51上使用的話,請把10級調光改成4級。謝謝合作!
* 74HC595 Q6 - 1602 EN
* 74HC595 Q5~Q2 - 1602 D5~D7(這一次硬件設計不合理,才只能搞軟件修正了。不過好處就是直接就可以洞洞板上對著聯,減少了燒壞595的概率。壞處就是CPU時間消耗的多了點。但是可以取消延時函數啦~
* 1602 D1~D4 直接懸空
* LCD_BACKLIGHT加一級PNP三極管射隨器加到液晶的A上,如果無需調光可以去掉這一部分電路和代碼
* 至于595的連接,只要不弄反SCL和SCK,就沒有問題啦~
*/
__sbit __at 0xE0 A_0;//方便位操作,硬件設置真的是敗筆 - -
__sbit __at 0xE1 A_1;
__sbit __at 0xE2 A_2;
__sbit __at 0xE3 A_3;
__sbit __at 0xE4 A_4;
__sbit __at 0xE5 A_5;
__sbit __at 0xE6 A_6;
__sbit __at 0xE7 A_7;
__sbit __at 0xF1 B_1;
__sbit __at 0xF2 B_2;
__sbit __at 0xF3 B_3;
__sbit __at 0xF4 B_4;
__sbit __at 0xF5 B_5;
__sbit __at 0xF6 B_6;
__sbit __at 0xF7 B_7;
unsigned char PWM_Cycle=0;//PWM占空比
unsigned char PWM_T=0;//PWM當前所處周期
unsigned char OutBuf=0xff;//輸出緩沖
#define LCD_OFF lcd_wcmd(0x08)
#define LCD_CUR lcd_wcmd(0x0E)
#define LCD_FCUR lcd_wcmd(0x0F)
void delay(unsigned int n)
{
unsigned int x,y;
for(x=n;x>0;x--)
for(y=100;y>0;y--);
}
void Send595()
{
unsigned char i,j;
j=0x80;
LCD1602_SCK=0;
for(i=0;i<8;i++)
{
LCD1602_SDA=OutBuf & j;
j=j>>1;
LCD1602_SCK=1;
LCD1602_SCK=0;
}
LCD1602_SDA=1;
LCD1602_SCL=0;
LCD1602_SCL=1;
}
void lcd_en()
{
//OutBuf=OutBuf|0x40;//E=1
B=OutBuf;
B_6=1;
OutBuf=B;
Send595();//輸出數據
B=OutBuf;
B_6=0;
OutBuf=B;//E=0
Send595();
}
/**********************************************************/
void lcd_wcmd(unsigned char i)
{
//寫命令
B=OutBuf;
B_7=0;//RS=0,RW=0;
OutBuf=B;
Send595();
//--------------
B=OutBuf;
ACC=i;
B_2=A_7;//D7=D7
B_3=A_6;//D6=D6
B_4=A_5;//D5=D5
B_5=A_4;//D4=D4
OutBuf=B;
Send595();
lcd_en();
//================
ACC=i;
B=OutBuf;
B_2=A_3;//D7=D3
B_3=A_2;//D6=D2
B_4=A_1;//D5=D1
B_5=A_0;//D4=D0
OutBuf=B;
Send595();
lcd_en();
}
void lcd_wdat(unsigned char i)
{
//寫數據
B=OutBuf;
B_7=1;//RS=0,RW=0;
OutBuf=B;
Send595();
//--------------
B=OutBuf;
ACC=i;
B_2=A_7;//D7=D7
B_3=A_6;//D6=D6
B_4=A_5;//D5=D5
B_5=A_4;//D4=D4
OutBuf=B;
Send595();
lcd_en();
//================
ACC=i;
B=OutBuf;
B_2=A_3;//D7=D3
B_3=A_2;//D6=D2
B_4=A_1;//D5=D1
B_5=A_0;//D4=D0
OutBuf=B;
Send595();
lcd_en();
}
void lcd_putchar(unsigned char addr,unsigned char ch)
{//寫一個字
lcd_wcmd(0x80+addr);
lcd_wdat(ch);
}
void lcd_display(unsigned char *l1,unsigned char *l2)
{//寫一屏幕
unsigned char i=0;
lcd_wcmd(0x80); //顯示地址設為80H(即00H,)上排第一位
for(i=0;i<16;i++)
{
lcd_wdat(l1[i]);
}
lcd_wcmd(0x80+0x40); //重新設定顯示地址為0xc0,即下排第1位
for(i=0;i<16;i++)
{
lcd_wdat(l2[i]);
}
}
void lcd_cls()
{
lcd_wcmd(0x01); //清屏延時函數可以省略
}
//米有讀函數,但是功能可以由上面的函數擴展哦~
void lcd_init()
{
OutBuf=0xff;
Send595();//復位VIO,這段代碼必須加。否則MCU忽然RESET會導致液晶亂碼
lcd_wcmd(0x30);//復位1602,這段代碼必須加,同上
lcd_en();//喚醒
lcd_wcmd(0x20);
lcd_en();//喚醒
lcd_wcmd(0x28); //四位,5x7
lcd_wcmd(0x0c); //開啟顯示屏,關光標,光標不閃爍
lcd_wcmd(0x06); //顯示地址遞增,即寫一個數據后,顯示位置右移一位
lcd_wcmd(0x01); //清屏
//調試的時候被設計失誤的對比度電位器坑死了:就是兩排小方塊死都不顯示
//查了半天都不知道
//最后捅了下電位器就正常了……
}
void lcd_print(unsigned char *str)
{
unsigned char addr=0x80,len=0;
lcd_wcmd(addr); //顯示地址設為80H(即00H,)上排第一位
while(*str!='\0')
{//等待末尾
if(len >= 16)
{
lcd_wcmd(0x80+0x40);//下排第一位,換行
len=0;
}
if(*str=='\n')
{
lcd_wcmd(0x80+0x40);//下排第一位,換行
}
else
{
lcd_wdat(*str);
}
str++;
len++;
}
}//寫字符串函數
void PWM() __interrupt 1 __using 1
{
ET0=0;
TF0=0;//CLR
TR0=0;
EA=0;
TH0=0xFC;TL0=0x00;//十級調光,頻率400Hz
if(PWM_T>10) {PWM_T=0;}
if(PWM_Cycle>PWM_T){ LCD1602_BACKLIGHT=0;}
if(PWM_Cycle<PWM_T) {LCD1602_BACKLIGHT=1;}
PWM_T++;
TR0=1;
ET0=1;
EA=1;
}
void main()
{//主函數
TMOD = 0x01 ;
TH0=0xFC;TL0=0x00;
TR0=1;ET0=1;
EA=1;//開中斷
//-----------------
lcd_init();
PWM_Cycle=10;//背光100%
//液晶初始化---------
while(1)
{
lcd_display(" Hello world! ","Serial mode 1602");
delay(2000);
lcd_display("Drive by 74HC595","Design by rgwan ");
delay(2000);
lcd_display("Website address:"," www.rwzy.co.cc ");
delay(2000);
lcd_display("Backlight ctl by"," T0 PWM 400Hz ");
delay(2000);
}
}
下面上圖片:

自己DIY的最小系統,旁邊是串口轉接器。這個板子上有兩個晶振跳線選擇
外加一個外界晶振。現在這個晶振跳在內置的11.0592上。

