微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > 为什么利用串口控制LCD1602 第一行的字符在第一行和第二行都显示啊

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

时间:10-02 整理:3721RD 点击:
代码在此
#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 != '\0')  //连续写入字符串数据,直到检测到结束符
    {
        LcdWriteDat(*str++);
    }
}
/* 初始化1602液晶 */
void InitLcd1602()
{
    LcdWriteCmd(0x38);  //16*2显示,5*7点阵,8位数据接口
    LcdWriteCmd(0x0C);  //显示器开,光标关闭
    LcdWriteCmd(0x06);  //文字不动,地址自动+1
    LcdWriteCmd(0x01);  //清屏
}
1602.c的代码

#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--);
}
主函数在此

http://bbs.elecfans.com/forum.php?mod=attachment&aid=Mjg0NTk3fDVhNDg5NjllMTc2MTAwNGIzNThkYTRiNmRlZTYyNzkxfDE1MDk2Mzk3MzU%3D&request=yes&_f=.jpgattach://284598.jpghttp://bbs.elecfans.com/forum.php?mod=attachment&aid=Mjg0NTk5fDA2ODU5YjcxOWMwYzQ1NjhiYWVlMDkyMTc2YzBmYzEzfDE1MDk2Mzk3MzU%3D&request=yes&_f=.jpg
现象在此




也就是我本来只想在第一行显示的素具也在第二行显示了 不知道为什么

程序有一个很明显的bug,在p[]和q[]数组的声明。unsigned char p[3]; unsigned char q[4];
你给p分配了3字节,q4字节,却没有给他们留下放字符串结束标志'\0'的空间,而LcdShowStr在显示时又是以'\0'作为结束标志。很明显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] = '\0';
q[4] = '\0';
后面两句不能省,哪怕未初始化的全局变量会默认被置为0。局部变量可不会

  LcdShowStr(5, 0, q);
LcdShowStr(5, 1, p);

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

dddddddddddddddddddddddddddddddddddd

不错的文档,顶一下

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top