求助:1602液晶查询忙标志的问题
时间:10-02
整理:3721RD
点击:
求助:为什么我做1602液晶模块实,采用查询忙标志的方法的话执行清屏幕指令后忙标志好像并没有置1?因为我的紧跟清屏幕指令的后一条指令老是得不到执行,直接被跳过去了。程序如下:(数据口接P0)
/******************************************************/
经仔细观察1602时序图后得知它所有的时序要求都是纳秒级,
而12M晶振时单片机的指令执行周期至少为1微秒,
故指令本身的执行时间已经满足了所有时序要求,
故例程中所有的nop延时语句均可删掉而不影响实验结果,
另,经实践证明,清屏指令后,1602好像并没有将忙状态置一,
明天换液晶试一下,暂不知为何,
/*********************************************************/
#include<reg52.h> //包含单片机寄存器的头文件
#include<intrins.h> //包含_nop_()函数定义的头文件
sbit RS=P2^0; //寄存器选择位,将RS位定义为P2.0引脚
sbit RW=P2^1; //读写选择位,将RW位定义为P2.1引脚
sbit E=P2^2; //使能信号位,将E位定义为P2.2引脚
sbit BF=P0^7; //忙碌标志位,,将BF位定义为P0.7引脚
unsigned char code string[ ]= {"WWW.AB.COM"};
unsigned char code string1[ ]={"QUICK STUDY MCU"};
unsigned char code User[]={0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00};//自定义字符℃ */
/*****************************************************
函数功能:延时1ms
***************************************************/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/*****************************************************
函数功能:延时若干毫秒
入口参数:n
***************************************************/
void delay(unsigned int n)
{
unsigned int i;
for(i=0;i<n;i++)
delay1ms();
}
/*****************************************************
函数功能:判断液晶模块的忙碌状态
返回值:result。result=1,忙碌;result=0,不忙
***************************************************/
bit BusyTest(void)
{
bit result;
RS=0; //根据规定,RS为低电平,RW为高电平时,可以读状态
RW=1;
E=1; //E=1,才允许读写
// _nop_(); //空操作
// _nop_();
// _nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
result=BF; //将忙碌标志电平赋给result
E=0;
return result;
}
/*****************************************************
函数功能:将模式设置指令或显示地址写入液晶模块
入口参数:dictate
***************************************************/
void Write_com (unsigned char dictate,bit n)
{
if(n)
{while(BusyTest()==1);} //如果忙就等待,n=1时才查询忙标志
RS=0; //根据规定,RS和R/W同时为低电平时,可以写入指令
RW=0;
E=0; //E置低电平(写指令时,
// 就是让E从0到1发生正跳变,所以应先置"0"
//_nop_();
//_nop_(); //空操作两个机器周期,给硬件反应时间
P0=dictate; //将数据送入P0口,即写入指令或地址
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=1; //E置高电平
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=0; //当E由高电平跳变成低电平时,液晶模块开始执行命令
}
/*****************************************************
函数功能:指定字符显示的实际地址
入口参数:x
***************************************************/
void WriteAddress(unsigned char x)
{
Write_com(x|0x80,1); //显示位置的确定方法规定为"80H+地址码x"
}
/*****************************************************
函数功能:将数据(字符的标准ASCII码)写入液晶模块
入口参数:y(为字符常量)
***************************************************/
void WriteData(unsigned char y)
{
while(BusyTest()==1);
RS=1; //RS为高电平,RW为低电平时,可以写入数据
RW=0;
E=0; //E置低电平(写指令时,,
// 就是让E从0到1发生正跳变,所以应先置"0"
P0=y; //将数据送入P0口,即将数据写入液晶模块
//_nop_();
//_nop_();
//_nop_();
// _nop_(); //空操作四个机器周期,给硬件反应时间
E=1; //E置高电平
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=0; //当E由高电平跳变成低电平时,液晶模块开始执行命令
}
/*****************************************************
函数功能:对LCD的显示模式进行初始化设置
***************************************************/
void LcdInt(void)
{
delay(15); //延时15ms,首次写指令时应给LCD一段较长的反应时间
Write_com(0x38,0); //显示模式设置:16×2显示,5×7点阵,8位数据接口
delay(5); //延时5ms
Write_com(0x38,0);
delay(5);
Write_com(0x38,0); //3次写 设置模式
delay(5);
Write_com(0x0c,1); //显示模式设置:显示开,wu光标,光标bu闪烁
delay(5);
Write_com(0x04,1); //输入模式设置:光标右移,字符不移
delay(5);
Write_com(0x01,1); //清屏幕指令,将以前的显示内容清除
delay(5);
}
void main(void) //主函数
{
unsigned char i;
LcdInt(); //调用LCD初始化函数
while(1)
{
Write_com(0x01,1);//清显示:清屏幕指令. 问题就出在这里,只要有这条清屏幕指令,下面紧跟的这个写地址的指令就执行不了, //必须加至少2毫秒延时后才可以顺利执行下条写DDRAM地址指令语句
WriteAddress(0x05); // 设置显示位置为第一行的第6个字。如果上面那条清屏语句存在,则这条老是得不到执行?为什么?
i = 0;
while(string != '\0') //'\0'是数组结束标志
{ // 显示字符 WWW.AB.COM
WriteData(string);
i++;
delay(100);
}
WriteAddress(0x40); // 设置显示位置为第二行的第1个字
i = 0;
while(string1 != '\0') //'\0'是数组结束标志
{ // 显示字符 QUICK STUDY MCU
WriteData(string1);
i++;
delay(100);
}
while(1);
}
}
/******************************************************/
经仔细观察1602时序图后得知它所有的时序要求都是纳秒级,
而12M晶振时单片机的指令执行周期至少为1微秒,
故指令本身的执行时间已经满足了所有时序要求,
故例程中所有的nop延时语句均可删掉而不影响实验结果,
另,经实践证明,清屏指令后,1602好像并没有将忙状态置一,
明天换液晶试一下,暂不知为何,
/*********************************************************/
#include<reg52.h> //包含单片机寄存器的头文件
#include<intrins.h> //包含_nop_()函数定义的头文件
sbit RS=P2^0; //寄存器选择位,将RS位定义为P2.0引脚
sbit RW=P2^1; //读写选择位,将RW位定义为P2.1引脚
sbit E=P2^2; //使能信号位,将E位定义为P2.2引脚
sbit BF=P0^7; //忙碌标志位,,将BF位定义为P0.7引脚
unsigned char code string[ ]= {"WWW.AB.COM"};
unsigned char code string1[ ]={"QUICK STUDY MCU"};
unsigned char code User[]={0x10,0x06,0x09,0x08,0x08,0x09,0x06,0x00};//自定义字符℃ */
/*****************************************************
函数功能:延时1ms
***************************************************/
void delay1ms()
{
unsigned char i,j;
for(i=0;i<10;i++)
for(j=0;j<33;j++)
;
}
/*****************************************************
函数功能:延时若干毫秒
入口参数:n
***************************************************/
void delay(unsigned int n)
{
unsigned int i;
for(i=0;i<n;i++)
delay1ms();
}
/*****************************************************
函数功能:判断液晶模块的忙碌状态
返回值:result。result=1,忙碌;result=0,不忙
***************************************************/
bit BusyTest(void)
{
bit result;
RS=0; //根据规定,RS为低电平,RW为高电平时,可以读状态
RW=1;
E=1; //E=1,才允许读写
// _nop_(); //空操作
// _nop_();
// _nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
result=BF; //将忙碌标志电平赋给result
E=0;
return result;
}
/*****************************************************
函数功能:将模式设置指令或显示地址写入液晶模块
入口参数:dictate
***************************************************/
void Write_com (unsigned char dictate,bit n)
{
if(n)
{while(BusyTest()==1);} //如果忙就等待,n=1时才查询忙标志
RS=0; //根据规定,RS和R/W同时为低电平时,可以写入指令
RW=0;
E=0; //E置低电平(写指令时,
// 就是让E从0到1发生正跳变,所以应先置"0"
//_nop_();
//_nop_(); //空操作两个机器周期,给硬件反应时间
P0=dictate; //将数据送入P0口,即写入指令或地址
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=1; //E置高电平
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=0; //当E由高电平跳变成低电平时,液晶模块开始执行命令
}
/*****************************************************
函数功能:指定字符显示的实际地址
入口参数:x
***************************************************/
void WriteAddress(unsigned char x)
{
Write_com(x|0x80,1); //显示位置的确定方法规定为"80H+地址码x"
}
/*****************************************************
函数功能:将数据(字符的标准ASCII码)写入液晶模块
入口参数:y(为字符常量)
***************************************************/
void WriteData(unsigned char y)
{
while(BusyTest()==1);
RS=1; //RS为高电平,RW为低电平时,可以写入数据
RW=0;
E=0; //E置低电平(写指令时,,
// 就是让E从0到1发生正跳变,所以应先置"0"
P0=y; //将数据送入P0口,即将数据写入液晶模块
//_nop_();
//_nop_();
//_nop_();
// _nop_(); //空操作四个机器周期,给硬件反应时间
E=1; //E置高电平
//_nop_();
//_nop_();
//_nop_();
//_nop_(); //空操作四个机器周期,给硬件反应时间
E=0; //当E由高电平跳变成低电平时,液晶模块开始执行命令
}
/*****************************************************
函数功能:对LCD的显示模式进行初始化设置
***************************************************/
void LcdInt(void)
{
delay(15); //延时15ms,首次写指令时应给LCD一段较长的反应时间
Write_com(0x38,0); //显示模式设置:16×2显示,5×7点阵,8位数据接口
delay(5); //延时5ms
Write_com(0x38,0);
delay(5);
Write_com(0x38,0); //3次写 设置模式
delay(5);
Write_com(0x0c,1); //显示模式设置:显示开,wu光标,光标bu闪烁
delay(5);
Write_com(0x04,1); //输入模式设置:光标右移,字符不移
delay(5);
Write_com(0x01,1); //清屏幕指令,将以前的显示内容清除
delay(5);
}
void main(void) //主函数
{
unsigned char i;
LcdInt(); //调用LCD初始化函数
while(1)
{
Write_com(0x01,1);//清显示:清屏幕指令. 问题就出在这里,只要有这条清屏幕指令,下面紧跟的这个写地址的指令就执行不了, //必须加至少2毫秒延时后才可以顺利执行下条写DDRAM地址指令语句
WriteAddress(0x05); // 设置显示位置为第一行的第6个字。如果上面那条清屏语句存在,则这条老是得不到执行?为什么?
i = 0;
while(string != '\0') //'\0'是数组结束标志
{ // 显示字符 WWW.AB.COM
WriteData(string);
i++;
delay(100);
}
WriteAddress(0x40); // 设置显示位置为第二行的第1个字
i = 0;
while(string1 != '\0') //'\0'是数组结束标志
{ // 显示字符 QUICK STUDY MCU
WriteData(string1);
i++;
delay(100);
}
while(1);
}
}
我记得没错的话,对1602的每一次操作你都测一下忙与否,这样比较保险。我以前也遇到过类似问题。