只需要2根線就能驅動1602的方法 1、單片機通過IIC與PCF8574進行通信。 寫好IIC通信程序(可以查找) 2、單片機4線制驅動1602 其中P0口的高四位接到LCD1602的DB7-DB4,P3.0-P3.1分別接RS,R/W,E; 下面給出驅動源程序(2個程序僅供參考有些許沒完善請自行修改) 4線制驅動方式需要7個IO口(RS、RW、E和4條數據線),而PCF8574提供了8位準雙向口,所以管腳還有剩余。 2線驅動電路圖
圖片6.png (24.39 KB, 下載次數: 40)
下載附件
2019-8-1 10:40 上傳
效果圖
圖片2.png (861.68 KB, 下載次數: 33)
下載附件
2019-8-1 10:40 上傳
實物圖
圖片3.png (462.55 KB, 下載次數: 36)
下載附件
2019-8-1 10:40 上傳
圖片4.png (511.27 KB, 下載次數: 32)
下載附件
2019-8-1 10:40 上傳
圖片5.png (738.42 KB, 下載次數: 32)
下載附件
2019-8-1 10:40 上傳
Proteus仿真
圖片1.png (19.73 KB, 下載次數: 30)
下載附件
2019-8-1 10:40 上傳
程序 #include <reg52.h> #include <intrins.h> sbit SCL = P3^0; sbit SDA = P3^1; bit ack; unsigned char LCD_data; unsigned char code digit[ ]={"0123456789"}; //定義字符數組顯示數字 //*****************延時************************ void delay_nus(unsigned int n) //N us延時函數 { unsigned int i=0; for (i=0;i<n;i++) _nop_(); } void delay_nms(unsigned int n) //N ms延時函數 { unsigned int i,j; for (i=0;i<n;i++) for (j=0;j<1140;j++); } void nop4() { _nop_(); //等待一個機器周期 _nop_(); //等待一個機器周期 _nop_(); //等待一個機器周期 _nop_(); //等待一個機器周期 } //*************************************** void Start() { SDA=1; _nop_(); SCL=1; nop4(); SDA=0; nop4(); SCL=0; _nop_(); _nop_(); } void Stop() { SDA=0; _nop_(); SCL=0; nop4();//>4us后SCL跳變 SCL=1; nop4(); SDA=1; _nop_(); _nop_(); } //****************************************** void Write_A_Byte(unsigned char c) { unsigned char BitCnt; for(BitCnt=0;BitCnt<8;BitCnt++) //要傳送的數據長度為8位 { if((c<<BitCnt)&0x80) SDA=1; //判斷發送位 else SDA=0; _nop_(); SCL=1; //置時鐘線為高,通知被控器開始接收數據位 nop4(); _nop_(); SCL=0; } _nop_(); _nop_(); SDA=1; //8位發送完后釋放數據線,準備接收應答位 _nop_(); _nop_(); SCL=1; _nop_(); _nop_(); _nop_(); if(SDA==1)ack=0; else ack=1; //判斷是否接收到應答信號 SCL=0; _nop_(); _nop_(); } bit Write_Random_Address_Byte(unsigned char add,unsigned char dat) { Start(); //啟動總線 Write_A_Byte(add); //發送器件地址 if(ack==0)return(0); Write_A_Byte(dat); //發送數據 if(ack==0)return(0); Stop(); //結束總線 return(1); } //********************液晶屏使能********************* void Enable_LCD_write() { LCD_data|=(1<<(3-1));//E=1; Write_Random_Address_Byte(0x40,LCD_data); delay_nus(2); LCD_data&=~(1<<(3-1));//E=0; Write_Random_Address_Byte(0x40,LCD_data); } //*************寫命令**************************** void LCD_write_command(unsigned char command) { delay_nus(16); LCD_data&=~(1<<(1-1));//RS=0; LCD_data&=~(1<<(2-1));//RW=0; Write_Random_Address_Byte(0x40,LCD_data); LCD_data&=0X0f; //清高四位 LCD_data|=command & 0xf0; //寫高四位 Write_Random_Address_Byte(0x40,LCD_data); Enable_LCD_write(); command=command<<4; //低四位移到高四位 LCD_data&=0x0f; //清高四位 LCD_data|=command&0xf0; //寫低四位 Write_Random_Address_Byte(0x40,LCD_data); Enable_LCD_write(); } //*************寫數據**************************** void LCD_write_data(unsigned char value) { delay_nus(16); LCD_data|=(1<<(1-1));//RS=1; LCD_data&=~(1<<(2-1));//RW=0; Write_Random_Address_Byte(0x40,LCD_data); LCD_data&=0X0f; //清高四位 LCD_data|=value&0xf0; //寫高四位 Write_Random_Address_Byte(0x40,LCD_data); Enable_LCD_write(); value=value<<4; //低四位移到高四位 LCD_data&=0x0f; //清高四位 LCD_data|=value&0xf0; //寫低四位 Write_Random_Address_Byte(0x40,LCD_data); Enable_LCD_write(); } //**********************設置顯示位置********************************* void set_position(unsigned char x,unsigned char y) { unsigned char position; if (y == 0) position = 0x80 + x; else position = 0xc0 + x; LCD_write_command(position); } //**********************顯示字符串***************************** void display_string(unsigned char x,unsigned char y,unsigned char *s) { set_position(x,y); while (*s) { LCD_write_data(*s); s++; } } //*************液晶初始化**************************** void LCD_init(void) { LCD_write_command(0x28); delay_nus(40); LCD_write_command(0x28); delay_nus(40); Enable_LCD_write(); delay_nus(40); LCD_write_command(0x28); //4位顯示!!!!!!!!!!!!!!!!!! LCD_write_command(0x0c); //顯示開 LCD_write_command(0x01); //清屏 delay_nms(2); } void main(void) { LCD_init(); display_string(4,0,"imxuheng"); //顯示一段文字 display_string(2,1,"Hello Today!"); //顯示一段文字 while(1); }
|