标题: 51单片机任意2个IO口(2线制)驱动LCD1602(包含4线制编程提供参考) [打印本页]
作者: 羽一生 时间: 2019-8-1 10:54
标题: 51单片机任意2个IO口(2线制)驱动LCD1602(包含4线制编程提供参考)
只需要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线驱动电路图
效果图
实物图
Proteus仿真
程序
#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);
}
作者: zxy2266 时间: 2019-8-13 15:26
请教地址是0x40吗?我的好像是3f,还有就是1602都可以4为传输吧?
作者: wanggf911 时间: 2019-8-13 20:55
很好,看了收益很深
作者: zxy2266 时间: 2019-8-14 21:22
搞不定,楼主还忙着那?
欢迎光临 (http://www.51hei.com/bbs/) |
Powered by Discuz! X3.1 |