为什么利用串口控制LCD1602 第一行的字符在第一行和第二行都显示啊
#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
不错的文档,顶一下