lcd1602液晶显示的程序
最近在看lcd的显示原理,代码有很多地方不理解,下面是我在网上找到一个显示代码:仿真中显示的是welcome!,还望大家能给我解释下(能解释多少是多少),谢谢了!#inc...
最近在看lcd的显示原理,代码有很多地方不理解,下面是我在网上找到一个显示代码:仿真中显示的是welcome!,还望大家能给我解释下(能解释多少是多少),谢谢了!
#include<reg52.h>
#include<intrins.h>
sbit RS=P2^0;
sbit RW=P2^1;
sbit E=P2^2;
sbit BF=P1^7;
char world[]="Welcome!";
char world1[]="Good morning!";
void delay1ms()
{
unsignedchar i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++);
}
void delay(unsigned char n)
{ unsignedchar i;
for(i=0;i<n;i++)
delay1ms();}
void nop(unsigned s)
{ unsignedchar i;
for(i=0;i<s;i++)
_nop_();}
unsigned char busytest(void)//忙绿检测中RS.RW.E为什么这样设置?为什么判断P1^7就能检测是否忙绿
{
bitresult;
RS=0;
RW=1;
E=1;
nop(4);
result=BF;
E=0;
returnresult;
}
void writecontrol(unsigned condata)//这个函数我也不懂?
{
while(busytest()==1);
RS=0;
RW=0;
E=0;
nop(2);
P1=condata;
nop(4);
E=1;
nop(4);
E=0;
}
void writedata(unsigned d)
{
while(busytest()==1);
RS=1;
RW=0;
E=0;
P1=d;
nop(4);//为什么函数中间加很多的空操作?
E=1;
nop(4);
E=0;
}
void lcdinit(void)//初始化,怎么这么多的延时,还有三个writecontrol(0x38)?
{
delay(15);
writecontrol(0x38);//8bit;16*2;5*7
delay(5);
writecontrol(0x38);
delay(5);
writecontrol(0x38);
delay(5);
writecontrol(0x0f);
delay(5);
writecontrol(0x06);
delay(5);
writecontrol(0x01);
delay(5);
}
void main()//lcd显示和数码管动态显示一样吗?不停的显示!
{
unsignedchar i;
lcdinit();
delay(10);
while(1)
{
writecontrol(0x01);
writecontrol(0x80);
i=0;
while(world[i]!='\0')
{
writedata(world[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
writecontrol(0xc0);
i=0;
while(world1[i]!='\0')
{
writedata(world1[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
}
} 展开
#include<reg52.h>
#include<intrins.h>
sbit RS=P2^0;
sbit RW=P2^1;
sbit E=P2^2;
sbit BF=P1^7;
char world[]="Welcome!";
char world1[]="Good morning!";
void delay1ms()
{
unsignedchar i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++);
}
void delay(unsigned char n)
{ unsignedchar i;
for(i=0;i<n;i++)
delay1ms();}
void nop(unsigned s)
{ unsignedchar i;
for(i=0;i<s;i++)
_nop_();}
unsigned char busytest(void)//忙绿检测中RS.RW.E为什么这样设置?为什么判断P1^7就能检测是否忙绿
{
bitresult;
RS=0;
RW=1;
E=1;
nop(4);
result=BF;
E=0;
returnresult;
}
void writecontrol(unsigned condata)//这个函数我也不懂?
{
while(busytest()==1);
RS=0;
RW=0;
E=0;
nop(2);
P1=condata;
nop(4);
E=1;
nop(4);
E=0;
}
void writedata(unsigned d)
{
while(busytest()==1);
RS=1;
RW=0;
E=0;
P1=d;
nop(4);//为什么函数中间加很多的空操作?
E=1;
nop(4);
E=0;
}
void lcdinit(void)//初始化,怎么这么多的延时,还有三个writecontrol(0x38)?
{
delay(15);
writecontrol(0x38);//8bit;16*2;5*7
delay(5);
writecontrol(0x38);
delay(5);
writecontrol(0x38);
delay(5);
writecontrol(0x0f);
delay(5);
writecontrol(0x06);
delay(5);
writecontrol(0x01);
delay(5);
}
void main()//lcd显示和数码管动态显示一样吗?不停的显示!
{
unsignedchar i;
lcdinit();
delay(10);
while(1)
{
writecontrol(0x01);
writecontrol(0x80);
i=0;
while(world[i]!='\0')
{
writedata(world[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
writecontrol(0xc0);
i=0;
while(world1[i]!='\0')
{
writedata(world1[i]);
i++;
delay(150);
}
for(i=0;i<4;i++)
delay(250);
}
} 展开
1个回答
展开全部
①问,RS,RW,E 的设置为 读状态时序 的过程,读出来的数据(数据口P1) 的最高位 刚好即为 液晶的 忙碌 标志位。
②问,writecontrol(unsigned condata) 函数 是个 写指令 函数;
③问,空操作 是为了让数据稳定后,才使能 液晶接收数据;
④问,写指令有时间 间隔要求,太频繁的读写会丢码,所以要加延时,三条38指令,是因为安全起见,上电后马上初始化会出现液晶电源不稳而丢码;
⑤问,液晶显示 只在 更新数据 的时候刷新一次即可,液晶会维持内容,无需重复刷新。
⑥附送,③和④问,都是驱动太恶心而造成的结果,好的驱动程序不需如此操作。液晶的时序要求为ns级,单片机的指令周期普遍为us级,根本不需过多累赘。
#include"reg52.h" //包含52头文件
#include"SMC1602A.h" //包含SMC1602A宏定义文件
#define BusyReadCount 10 //读忙标志等待次数
#define SMC1602_Data P0 //定义 数据接口
//sbit SMC1602_VO=P2^4; //定义 VO对比度接口
sbit SMC1602_RW=P2^5; //定义 R/W接口25
sbit SMC1602_RS=P2^6; //定义 RS接口26
sbit SMC1602_E=P2^7; //定义 E接口27
#define SMC1602_En SMC1602_E=1 //使能
#define SMC1602_Dis SMC1602_E=0 //禁止
uchar SMC1602_Read(bit read_type) //1602液晶屏读函数
{
uchar read_data;
SMC1602_Dis; //禁止使能
SMC1602_RW=ReadOperate; //读操作
SMC1602_RS=read_type; //读类型:0状态,1数据
SMC1602_En; //开启使能
read_data=SMC1602_Data; //存储结果
SMC1602_Dis; //禁止使能
return read_data; //返回结果
}
void SMC1602_WriteByte(bit write_type,uchar write_data) //1602液晶屏读函数
{
uchar i=BusyReadCount;
for(;i;i--); //延时操作,为写操作预留回复时间
while((SMC1602_Read(CommOperate)&BusyState) &&(++i<=BusyReadCount)); //读取忙标志(BusyReadCount次),若均忙中,则不再读取忙标志,直接执行写操作
//while(SMC1602_Read(CommOperate)&BusyState) if(++i>BusyReadCount) return; //读取忙标志,若BusyReadCount次均忙中,则不进行写操作
//while(SMC1602_Read(CommOperate)&BusyState); //等待空闲(死等)
SMC1602_Dis; //禁止使能
SMC1602_RW=WriteOperate; //写操作
SMC1602_RS=write_type; //写类型:0指令,1数据
SMC1602_Data=write_data; //写操作,将操作数送的数据口
SMC1602_En; //开启使能
SMC1602_Dis; //禁止使能
}
void SMC1602_WriteCGRAM(uchar *write_buf,uchar start_loca,uchar word_num,uchar start_addr) //SMC1602写CGRAM函数,用于自定义字符
{
uchar i,j;
write_buf+=start_loca; //指向"需写入数据数组"的起始位置
SMC1602_WriteByte(CommOperate,CGRAMAddr|start_addr<<3); //写CGRAM操作,并将CGRAM起始地址设为 start_addr
for(j=0;j<word_num;j++) //自定义字符数量
for(i=0;i<8;i++) SMC1602_WriteByte(DataOperate,*write_buf++); //写入一个自定义字符8个字节数据
}
void SMC1602_Init() //1602液晶屏初始化函数
{
uint i;
SMC1602_WriteByte(CommOperate,DisplayMode); //显示模式设置:16×2显示,5×7点阵,8位数据接口
SMC1602_WriteByte(CommOperate,ScreenMode|ScreenOn); //光标模式设置:开启整体显示,开启光标显示,开启光标闪烁
SMC1602_WriteByte(CommOperate,InputMode); //输入方式设置:关闭整屏移动,开启光标正移动(+1)
SMC1602_WriteByte(CommOperate,CleanLCD); //清屏,复位光标
SMC1602_WriteByte(CommOperate,FirstCol); //定位第一行
for(i=150;i;i--); //等待电源稳定,否则写CGRAM数据(自定义字符)时容易丢失,uint执行周期长,用uchar将会缩短时间,不足以稳定LCD
}
②问,writecontrol(unsigned condata) 函数 是个 写指令 函数;
③问,空操作 是为了让数据稳定后,才使能 液晶接收数据;
④问,写指令有时间 间隔要求,太频繁的读写会丢码,所以要加延时,三条38指令,是因为安全起见,上电后马上初始化会出现液晶电源不稳而丢码;
⑤问,液晶显示 只在 更新数据 的时候刷新一次即可,液晶会维持内容,无需重复刷新。
⑥附送,③和④问,都是驱动太恶心而造成的结果,好的驱动程序不需如此操作。液晶的时序要求为ns级,单片机的指令周期普遍为us级,根本不需过多累赘。
#include"reg52.h" //包含52头文件
#include"SMC1602A.h" //包含SMC1602A宏定义文件
#define BusyReadCount 10 //读忙标志等待次数
#define SMC1602_Data P0 //定义 数据接口
//sbit SMC1602_VO=P2^4; //定义 VO对比度接口
sbit SMC1602_RW=P2^5; //定义 R/W接口25
sbit SMC1602_RS=P2^6; //定义 RS接口26
sbit SMC1602_E=P2^7; //定义 E接口27
#define SMC1602_En SMC1602_E=1 //使能
#define SMC1602_Dis SMC1602_E=0 //禁止
uchar SMC1602_Read(bit read_type) //1602液晶屏读函数
{
uchar read_data;
SMC1602_Dis; //禁止使能
SMC1602_RW=ReadOperate; //读操作
SMC1602_RS=read_type; //读类型:0状态,1数据
SMC1602_En; //开启使能
read_data=SMC1602_Data; //存储结果
SMC1602_Dis; //禁止使能
return read_data; //返回结果
}
void SMC1602_WriteByte(bit write_type,uchar write_data) //1602液晶屏读函数
{
uchar i=BusyReadCount;
for(;i;i--); //延时操作,为写操作预留回复时间
while((SMC1602_Read(CommOperate)&BusyState) &&(++i<=BusyReadCount)); //读取忙标志(BusyReadCount次),若均忙中,则不再读取忙标志,直接执行写操作
//while(SMC1602_Read(CommOperate)&BusyState) if(++i>BusyReadCount) return; //读取忙标志,若BusyReadCount次均忙中,则不进行写操作
//while(SMC1602_Read(CommOperate)&BusyState); //等待空闲(死等)
SMC1602_Dis; //禁止使能
SMC1602_RW=WriteOperate; //写操作
SMC1602_RS=write_type; //写类型:0指令,1数据
SMC1602_Data=write_data; //写操作,将操作数送的数据口
SMC1602_En; //开启使能
SMC1602_Dis; //禁止使能
}
void SMC1602_WriteCGRAM(uchar *write_buf,uchar start_loca,uchar word_num,uchar start_addr) //SMC1602写CGRAM函数,用于自定义字符
{
uchar i,j;
write_buf+=start_loca; //指向"需写入数据数组"的起始位置
SMC1602_WriteByte(CommOperate,CGRAMAddr|start_addr<<3); //写CGRAM操作,并将CGRAM起始地址设为 start_addr
for(j=0;j<word_num;j++) //自定义字符数量
for(i=0;i<8;i++) SMC1602_WriteByte(DataOperate,*write_buf++); //写入一个自定义字符8个字节数据
}
void SMC1602_Init() //1602液晶屏初始化函数
{
uint i;
SMC1602_WriteByte(CommOperate,DisplayMode); //显示模式设置:16×2显示,5×7点阵,8位数据接口
SMC1602_WriteByte(CommOperate,ScreenMode|ScreenOn); //光标模式设置:开启整体显示,开启光标显示,开启光标闪烁
SMC1602_WriteByte(CommOperate,InputMode); //输入方式设置:关闭整屏移动,开启光标正移动(+1)
SMC1602_WriteByte(CommOperate,CleanLCD); //清屏,复位光标
SMC1602_WriteByte(CommOperate,FirstCol); //定位第一行
for(i=150;i;i--); //等待电源稳定,否则写CGRAM数据(自定义字符)时容易丢失,uint执行周期长,用uchar将会缩短时间,不足以稳定LCD
}
更多追问追答
追问
谢谢你详细的解答,我收获不少。我还想问下,1.在初始化中writecontrol(0x38);是设置16*2显示,这个意思是该lcd是可以显示两行,每行16个字符吗?(可是lcd1602明明只是单行显示的啊) 2.在主函数while(1)中 writecontrol(0x80); writecontrol(0xc0);//为什么是第一行和第二行地址?谢谢!
追答
①1602 的 意思就是 02 行,每行 16格(字符),1602可以单行显示,但它只是单行显示的想法是错误的。0x38的代码含义如下:
#define DisplayMode 0x38 //显示模式设置(001(DL) NFxx) DL=0/1:4位/8位数据接口,N=0/1:一行/两行显示,F=0/1:5×7点阵/5×10点阵字符;
②0x80 为第一行的起始地址,这是1602内部设计决定的;实际上,1602 内部的存储量,并不止看得见 的每行16格,而是每行40格,但显示器只能显示其中的16格;就像一间房子,里面的宽度是40分米,而开的窗口为 16分米,于是,人在外面只能看到里面 16分米的宽度。
40= 0x28= 0010 1000 b,即需要 6个bit 存储该值,而6个bit 的最大范围为 64(2的6次方,0x40),于是,第二行的地址就为:0x80(第一行地址)+0x40=0xC0;
#define FirstCol 0x80 //第一行首地址0x80-0xA7 (40Byte RAM,显示范围仅16Byte)
#define SecondCol 0xC0 //第二行首地址0xC0-0xE7 (40Byte RAM,显示范围仅16Byte)
推荐律师服务:
若未解决您的问题,请您详细描述您的问题,通过百度律临进行免费专业咨询