为什么利用串口控制LCD1602 第一行的字符在第一行和第二行都显示啊

2019-07-15 16:05发布

代码在此
#include <reg52.h>

#define LCD1602_DB  P0
sbit LCD1602_RS = P1^0;
sbit LCD1602_RW = P1^1;
sbit LCD1602_E  = P1^5;

/* 等待液晶准备好 */
void LcdWaitReady()
{
    unsigned char sta;

    LCD1602_DB = 0xFF;
    LCD1602_RS = 0;
    LCD1602_RW = 1;
    do {
        LCD1602_E = 1;
        sta = LCD1602_DB; //读取状态字
        LCD1602_E = 0;
    } while (sta & 0x80); //bit7等于1表示液晶正忙,重复检测直到其等于0为止
}
/* 向LCD1602液晶写入一字节命令,cmd-待写入命令值 */
void LcdWriteCmd(unsigned char cmd)
{
    LcdWaitReady();
    LCD1602_RS = 0;
    LCD1602_RW = 0;
    LCD1602_DB = cmd;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 向LCD1602液晶写入一字节数据,dat-待写入数据值 */
void LcdWriteDat(unsigned char dat)
{
    LcdWaitReady();
    LCD1602_RS = 1;
    LCD1602_RW = 0;
    LCD1602_DB = dat;
    LCD1602_E  = 1;
    LCD1602_E  = 0;
}
/* 设置显示RAM起始地址,亦即光标位置,(x,y)-对应屏幕上的字符坐标 */
void LcdSetCursor(unsigned char x, unsigned char y)
{
    unsigned char addr;

    if (y == 0)  //由输入的屏幕坐标计算显示RAM的地址
        addr = 0x00 + x;  //第一行字符地址从0x00起始
    else
        addr = 0x40 + x;  //第二行字符地址从0x40起始
    LcdWriteCmd(addr | 0x80);  //设置RAM地址
}
/* 在液晶上显示字符串,(x,y)-对应屏幕上的起始坐标,str-字符串指针 */
void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str)
{
    LcdSetCursor(x, y);   //设置起始地址
    while (*str != '')  //连续写入字符串数据,直到检测到结束符
    {
        LcdWriteDat(*str++);
    }
}
/* 初始化1602液晶 */
void InitLcd1602()
{
    LcdWriteCmd(0x38);  //16*2显示,5*7点阵,8位数据接口
    LcdWriteCmd(0x0C);  //显示器开,光标关闭
    LcdWriteCmd(0x06);  //文字不动,地址自动+1
    LcdWriteCmd(0x01);  //清屏
}

1602.c的代码
友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
该问题目前已经被作者或者管理员关闭, 无法添加新回复
8条回答
489609274
1楼-- · 2019-07-15 16:36

#include <reg52.h>
unsigned char T0RH = 0;  //T0重载值的高字节
unsigned char T0RL = 0;  //T0重载值的低字节
unsigned char RxdByte;  //串口接收到的字节
extern void InitLcd1602();

bit cnt=0;
unsigned char p[3];
unsigned char q[4];
void encode();
void ConfigTimer0(unsigned int ms);
void ConfigUART(unsigned int baud);
void LcdWriteDat(unsigned char dat);
extern void LcdShowStr(unsigned char x, unsigned char y, unsigned char *str);
void geton();
void getoff();
void getnrol();
void getlow();
void gethigh();
void main()
{
    EA = 1;       //使能总中断
        Start18B20();     //启动DS18B20
    ConfigUART(9600);  //配置波特率为9600
    InitLcd1602();    //初始化液晶
        LcdShowStr(0, 0, "MODE:     T:  `C");
    LcdShowStr(0, 1, "AnTo:          %");
    while(1)
        {
            GetShowTemp();
            if(cnt==1)
                {
                                 
                 LcdShowStr(5, 0, q);
         LcdShowStr(5, 1, p);
                 cnt = 0;
        }                          
         
   
}
}
/* 串口配置函数,baud-通信波特率 */
void ConfigUART(unsigned int baud)
{
    SCON  = 0x50;  //配置串口为模式1
    TMOD &= 0x0F;  //清零T1的控制位
    TMOD |= 0x20;  //配置T1为模式2
    TH1 = 256 - (11059200/12/32)/baud;  //计算T1重载值
    TL1 = TH1;     //初值等于重载值
    ET1 = 0;       //禁止T1中断
    ES  = 1;       //使能串口中断
    TR1 = 1;       //启动T1
}
void fanhui()
{
    if(RxdByte=='1')
{
     SBUF='a';
  }
}
/* UART中断服务函数 */
void InterruptUART() interrupt 4
{
         
    if (RI)  //接收到字节
    {
               
        RI = 0;  //手动清零接收中断标志位
        RxdByte = SBUF;  //接收到的数据保存到接收字节变量中
        encode() ;              
            SBUF = RxdByte;
                cnt=1;

    }
    if (TI)  //字节发送完毕
    {
        TI = 0;  //手动清零发送中断标志位
    }
}
void encode()
{
   switch(RxdByte)
   {
            case '1': geton();break;
         case '2': getoff();break;
         case '3': getlow();break;
         case '4': getnrol();break;
         case '5': gethigh();break;
         default:break;
   }
   
}
void geton()
{
p[0] = 'o';
p[1] = 'n';
p[2] =        ' ';
q[0] = 'l';
q[1] = 'o';
q[2] = 'w';
q[3] = ' ';
}
void getoff()
{
p[0] = 'o';
p[1] = 'f';
p[2] = 'f';
}
void getlow()
{
q[0] = 'l';
q[1] = 'o';
q[2] = 'w';
q[3] = ' ';

}
void getnrol()
{
q[0] = 'n';
q[1] = 'r';
q[2] = 'o';
q[3] = 'l';
}
void gethigh()
{
q[0] = 'h';
q[1] = 'i';
q[2] = 'g';
q[3] =        'h';
}
void delay(unsigned int z)       //延时函数                       
{
    unsigned int x,y;  
    for(x=z;x>0;x--)
        for(y=122;y>0;y--);
}
主函数在此
489609274
2楼-- · 2019-07-15 21:14
http://bbs.elecfans.com/forum.php?mod=attachment&aid=Mjg0NTk3fGNmMTA3MzQ2MDNjNzU0ZTA4OWEwNzk1YjMzZGJmZjA2fDE1NTY3NjM1MDY%3D&request=yes&_f=.jpgattach://284598.jpghttp://bbs.elecfans.com/forum.php?mod=attachment&aid=Mjg0NTk5fGNmNjk1ODQxNWY0MDhhYjg4MGQ0MzdhNmFlMTUxOTBkfDE1NTY3NjM1MDY%3D&request=yes&_f=.jpg
现象在此
IMG_20150917_213519.jpg
IMG_20150917_213609.jpg
IMG_20150917_213637.jpg
489609274
3楼-- · 2019-07-16 02:05
也就是我本来只想在第一行显示的素具也在第二行显示了 不知道为什么
1522495332
4楼-- · 2019-07-16 05:54
程序有一个很明显的bug,在p[]和q[]数组的声明。unsigned char p[3]; unsigned char q[4];
你给p分配了3字节,q4字节,却没有给他们留下放字符串结束标志''的空间,而LcdShowStr在显示时又是以''作为结束标志。很明显p和q都没有这个标志位。第3幅图的“high”跑到“off”后面,有可能是在RAM中q数组刚好是接着p数组后面放的。程序运行到LcdShowStr(5, 1, p);这句时,把p里面的"off"显示完后,LcdShowStr函数里的str指针变量指向了q数组,就把q数组里的“high”也显示出来。
不管你说的现象是不是这个原因,你都要把我指出来的bug解决掉,方法很简单:
unsigned char p[4];
unsigned char q[5];
p[3] = '';
q[4] = '';
后面两句不能省,哪怕未初始化的全局变量会默认被置为0。局部变量可不会
dabing89
5楼-- · 2019-07-16 07:49
  LcdShowStr(5, 0, q);
LcdShowStr(5, 1, p);
489609274
6楼-- · 2019-07-16 09:29
1522495332 发表于 2015-9-17 22:52
程序有一个很明显的bug,在p[]和q[]数组的声明。unsigned char p[3]; unsigned char q[4];
你给p分配了3字节,q4字节,却没有给他们留下放字符串结束标志''的空间,而LcdShowStr在显示时又是以''作为结束标志。很明显p和q都没有这个标志位。第3幅图的“high”跑到“off”后面,有可能是在RAM中q数组刚好是接着p数组后面 ...

多谢大哥了   回去一用果然好了

一周热门 更多>