单片机学习小组
直播中

陈小艳

7年用户 196经验值
私信 关注

为什么LCD1602不能同时显示温度和时间?

程序目标:1602第一行显示温度,第二行显示时间。温度用DS18B20,时间用DS1302。
目前存在问题:用1602单独显示温度和时间都没得问题。合在一起显示就出现温度值为0,时间可以正常显示。
初步怀疑是我的main函数写法不对。请大家帮忙分析一下有到底有什么问题。谢谢~  
硬件威廉希尔官方网站 :用的51单片机STC89C52,LCD1602,DS18B20,DS1302。后门附图。
以下为程序,只有.C文件,头文件未包含,全部程序附在打包文件里面:
一、主程序  main.c
/**************************************************************************************
*                              LCD1602液晶显示实验                                                                                                  *
实现现象:下载程序后插上LCD1602液晶在开发板上,即可显示
注意事项:如果不想让点阵模块显示,可以将74HC595模块上的JP595短接片拔掉。                                                                                                                                                                  
***************************************************************************************/
#include "reg52.h"                         //此文件中定义了单片机的一些特殊功能寄存器
#include "lcd.h"
#include "temp.h"
#include
#include "DS1302.H"
typedef unsigned int u16;          //对数据类型进行声明定义
typedef unsigned char u8;
//u8 Disp[]="0Pechin Scien0e ";
//u8 Disp[8];
u8 DisplayData[9];
u16 DisplayDatatime[9];
bit TempDs18b20Flag;
bit Ds1302Flag;
/*******************************************************************************
* 函 数 名         : delay
* 函数功能                   : 延时函数
*******************************************************************************/
void delay1ms(void)   //误差 0us 延时1ms
{
    unsigned char a,b;
    for(b=199;b>0;b--)
        for(a=1;a>0;a--);
}
void delayxms(u8 x)//延时X个ms
{
        unsigned char a,b;
        for(b=x;b>0;b--)
        {
                delay1ms();
        }
}
        
/*******************************************************************************
* 函 数 名         : dataprosTime()
* 函数功能                   : 时间读取处理转换函数
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void dataprosTime()         
{
           Ds1302ReadTime();
        DisplayDataTime[0] = TIME[2]/16+0x30;                                //时
        DisplayDataTime[1] = TIME[2]%16+0x30;                                 
        DisplayDataTime[2] = 0x2d;//0100 0000
        DisplayDataTime[3] = TIME[1]/16+0x30;                                //分
        DisplayDataTime[4] = TIME[1]%16+0x30;        
        DisplayDataTime[5] = 0x2d;
        DisplayDataTime[6] = TIME[0]/16+0x30;                                //秒
        DisplayDataTime[7] = TIME[0]%16+0x30;
}
/*******************************************************************************
* 函 数 名         : datapros()
* 函数功能                   : 温度读取处理转换函数
* 输    入         : temp
* 输    出         : 无
*******************************************************************************/
void datapros(int temp)         
{
        //int temp;        
        //temp=Ds18b20ReadTemp();
        
           float tp;  
        if(temp< 0)                                //当温度值为负数
          {
                DisplayData[0] = 0x2d;           //   - 0010 1101
                //因为读取的温度是实际温度的补码,所以减1,再取反求出原码
                temp=temp-1;
                temp=~temp;
                tp=temp;
                temp=tp*0.0625*100+0.5;        
                //留两个小数点就*100,+0.5是四舍五入,因为C语言浮点数转换为整型的时候把小数点
                //后面的数自动去掉,不管是否大于0.5,而+0.5之后大于0.5的就是进1了,小于0.5的就
                //算加上0.5,还是在小数点后面。
          }
         else
          {                        
                DisplayData[0] = 0x2b;//第0位数据,正温度第0位+号,00101011
                tp=temp;//因为数据处理有小数点所以将温度赋给一个浮点型变量
                //如果温度是正的那么,那么正数的原码就是补码它本身
                temp=tp*0.0625*100+0.5;        
                //留两个小数点就*100,+0.5是四舍五入,因为C语言浮点数转换为整型的时候把小数点
                //后面的数自动去掉,不管是否大于0.5,而+0.5之后大于0.5的就是进1了,小于0.5的就
                //算加上0.5,还是在小数点后面。
          }
                                
        DisplayData[1] = temp / 10000;//开发板左边数第三个数码管,前面扩大100倍,这里除以10000,取百位数字
        DisplayData[2] = temp % 10000 / 1000+0x30;//先取余数,再除以1000,取十位数字
        DisplayData[3] = temp % 1000 / 100+0x30;// | 0x80;//取个位数字,再加上小数点标识0x80
        DisplayData[4] = 0x2e;//小数点
               
        DisplayData[5] = temp % 100 / 10+0x30;//取小数点后1位
        DisplayData[6] = temp % 10+0x30;//取小数点后2位
                DisplayData[7] = 0xdf;//1101 1111 °
                DisplayData[8] = 0x43;//0100 0011 C
        
                if(DisplayData[1]==0)//如果温度没有上百,则显示空白位
                {
                        DisplayData[1]=0x20;//显示空白
                }
                else
                {
                        DisplayData[1] = temp / 10000+0x30;//上百则显示ASCII数字
                }        
               
}
/*******************************************************************************
* 函 数 名         : main
* 函数功能                   : 主函数
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void main(void)
{
                u8 i;        
        
        LcdInit();//LCD初始化
        
        Ds1302Init();//1302初始化
                        
        while(1)
        {        
               
                datapros(Ds18b20ReadTemp());//温度数据处理函数
                  LcdWriteCom(0x80); //显示第1行
                for(i=0;i<9;i++)
        {
                LcdWriteData(DisplayData[ i ]);        //温度数据送LCD显示
        }
        
/*························································································*/        
        
        dataprosTime();//DS1302时间数据处理
        LcdWriteCom(0xC0); //显示第2行
        
        for(i=0;i<8;i++)
        {
                LcdWriteData(DisplayDataTime[ i ]);        //时间数据送LCD显示
        }
               
        delayxms(100);//延时100ms        
        LcdWriteCom(0x02);  //撤回左上角  0000 0010
                        
                }
                        
}
/////////////////////////////////////////////////////////////////////////////////////////
二、驱动程序DS1302.C
#include "ds1302.h"
//---DS1302写入和读取时分秒的地址命令---//
//---秒分时日月周年 最低位读写位;-------//
uchar code READ_RTC_ADDR[7] = {0x81, 0x83, 0x85, 0x87, 0x89, 0x8b, 0x8d};
uchar code WRITE_RTC_ADDR[7] = {0x80, 0x82, 0x84, 0x86, 0x88, 0x8a, 0x8c};
//---DS1302时钟初始化2016年5月7日星期六12点00分00秒。---//
//---存储顺序是秒分时日月周年,存储格式是用BCD码---//
uchar TIME[7] = {0, 0, 0x12, 0x07, 0x05, 0x06, 0x16};
/*******************************************************************************
* 函 数 名         : Ds1302Write
* 函数功能                   : 向DS1302命令(地址+数据)
* 输    入         : addr,dat
* 输    出         : 无
*******************************************************************************/
void Ds1302Write(uchar addr, uchar dat)
{
        uchar n;
        RST = 0;
        _nop_();
        SCLK = 0;//先将SCLK置低电平。
        _nop_();
        RST = 1; //然后将RST(CE)置高电平。
        _nop_();
        for (n=0; n<8; n++)//开始传送八位地址命令
        {
                DSIO = addr & 0x01;//数据从低位开始传送
                addr >>= 1;
                SCLK = 1;//数据在上升沿时,DS1302读取数据
                _nop_();
                SCLK = 0;
                _nop_();
        }
        for (n=0; n<8; n++)//写入8位数据
        {
                DSIO = dat & 0x01;
                dat >>= 1;
                SCLK = 1;//数据在上升沿时,DS1302读取数据
                _nop_();
                SCLK = 0;
                _nop_();        
        }        
                 
        RST = 0;//传送数据结束
        _nop_();
}
/*******************************************************************************
* 函 数 名         : Ds1302Read
* 函数功能                   : 读取一个地址的数据
* 输    入         : addr
* 输    出         : dat
*******************************************************************************/
uchar Ds1302Read(uchar addr)
{
        uchar n,dat,dat1;
        RST = 0;
        _nop_();
        SCLK = 0;//先将SCLK置低电平。
        _nop_();
        RST = 1;//然后将RST(CE)置高电平。
        _nop_();
        for(n=0; n<8; n++)//开始传送八位地址命令
        {
                DSIO = addr & 0x01;//数据从低位开始传送
                addr >>= 1;
                SCLK = 1;//数据在上升沿时,DS1302读取数据
                _nop_();
                SCLK = 0;//DS1302下降沿时,放置数据
                _nop_();
        }
        _nop_();
        for(n=0; n<8; n++)//读取8位数据
        {
                dat1 = DSIO;//从最低位开始接收
                dat = (dat>>1) | (dat1<<7);
                SCLK = 1;
                _nop_();
                SCLK = 0;//DS1302下降沿时,放置数据
                _nop_();
        }
        RST = 0;
        _nop_();        //以下为DS1302复位的稳定时间,必须的。
        SCLK = 1;
        _nop_();
        DSIO = 0;
        _nop_();
        DSIO = 1;
        _nop_();
        return dat;        
}
/*******************************************************************************
* 函 数 名         : Ds1302Init
* 函数功能                   : 初始化DS1302.
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void Ds1302Init() //写入初始时间值
{
        uchar n;
        Ds1302Write(0x8E,0X00);                 //禁止写保护,就是关闭写保护功能
        for (n=0; n<7; n++)//写入7个字节的时钟信号:分秒时日月周年
        {
                Ds1302Write(WRITE_RTC_ADDR[n],TIME[n]);        
        }
        Ds1302Write(0x8E,0x80);                 //打开写保护功能
}
/*******************************************************************************
* 函 数 名         : Ds1302ReadTime
* 函数功能                   : 读取时钟信息
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void Ds1302ReadTime()
{
        uchar n;
        for (n=0; n<7; n++)//读取7个字节的时钟信号:分秒时日月周年
        {
                TIME[n] = Ds1302Read(READ_RTC_ADDR[n]);
        }
               
}
三、温度驱动程序 DS18B20.C
#include "temp.h"
/*******************************************************************************
* 函 数 名         : Delay1ms
* 函数功能                   : 延时函数
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void Delay1ms(uint y)
{
        uint x;
        for( ; y>0; y--)
        {
                for(x=110; x>0; x--);
        }
}
/*******************************************************************************
* 函 数 名         : Ds18b20Init
* 函数功能                   : 初始化
* 输    入         : 无
* 输    出         : 初始化成功返回1,失败返回0
*******************************************************************************/
uchar Ds18b20Init()
{
        uchar i;
        DSPORT = 0;                         //将总线拉低480us~960us
        i = 70;        
        while(i--);//延时642us
        DSPORT = 1;                        //然后拉高总线,如果DS18B20做出反应会将在15us~60us后总线拉低
        i = 0;
        while(DSPORT)        //等待DS18B20拉低总线
        {
                Delay1ms(1);
                i++;
                if(i>5)//等待>5MS
                {
                        return 0;//初始化失败
                }
        
        }
        return 1;//初始化成功
}
/*******************************************************************************
* 函 数 名         : Ds18b20WriteByte
* 函数功能                   : 向18B20写入一个字节
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void Ds18b20WriteByte(uchar dat)
{
        uint i, j;
        for(j=0; j<8; j++)
        {
                DSPORT = 0;                       //每写入一位数据之前先把总线拉低1us
                i++;
                DSPORT = dat & 0x01;  //然后写入一个数据,从最低位开始
                i=6;
                while(i--); //延时68us,持续时间最少60us
                DSPORT = 1;        //然后释放总线,至少1us给总线恢复时间才能接着写入第二个数值
                dat >>= 1;
        }
}
/*******************************************************************************
* 函 数 名         : Ds18b20ReadByte
* 函数功能                   : 读取一个字节
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
uchar Ds18b20ReadByte()
{
        uchar byte, bi;
        uint i, j;        
        for(j=8; j>0; j--)
        {
                DSPORT = 0;//先将总线拉低1us
                i++;
                DSPORT = 1;//然后释放总线
                i++;
                i++;//延时6us等待数据稳定
                bi = DSPORT;         //读取数据,从最低位开始读取
                /*将byte左移一位,然后与上右移7位后的bi,注意移动之后移掉那位补0。*/
                byte = (byte >> 1) | (bi << 7);                                                  
                i = 4;                //读取完之后等待48us再接着读取下一个数
                while(i--);
        }                                
        return byte;
}
/*******************************************************************************
* 函 数 名         : Ds18b20ChangTemp
* 函数功能                   : 让18b20开始转换温度
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void  Ds18b20ChangTemp()
{
        Ds18b20Init();
        Delay1ms(1);
        Ds18b20WriteByte(0xcc);                //跳过ROM操作命令                 
        Ds18b20WriteByte(0x44);            //温度转换命令
        //Delay1ms(100);        //等待转换成功,而如果你是一直刷着的话,就不用这个延时了
}
/*******************************************************************************
* 函 数 名         : Ds18b20ReadTempCom
* 函数功能                   : 发送读取温度命令
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
void  Ds18b20ReadTempCom()
{        
        Ds18b20Init();
        Delay1ms(1);
        Ds18b20WriteByte(0xcc);         //跳过ROM操作命令
        Ds18b20WriteByte(0xbe);         //发送读取温度命令
}
/*******************************************************************************
* 函 数 名         : Ds18b20ReadTemp
* 函数功能                   : 读取温度
* 输    入         : 无
* 输    出         : 无
*******************************************************************************/
int Ds18b20ReadTemp()
{
        int temp = 0;
        uchar tmh, tml;
        Ds18b20ChangTemp();                                 //先写入转换命令
        Ds18b20ReadTempCom();                        //然后等待转换完后发送读取温度命令
        tml = Ds18b20ReadByte();                //读取温度值共16位,先读低字节
        tmh = Ds18b20ReadByte();                //再读高字节
        temp = tmh;
        temp <<= 8;
        temp |= tml;
        return temp;
}
四、LCD驱动程序 1602.C
#include "lcd.h"
/*******************************************************************************
* 函 数 名         : Lcd1602_Delay1ms
* 函数功能                   : 延时函数,延时1ms
* 输    入         : c
* 输    出         : 无
* 说    名         : 该函数是在12MHZ晶振下,12分频单片机的延时。
*******************************************************************************/
void Lcd1602_Delay1ms(uint c)   //误差 0us  1ms延时
{
    uchar a,b;
        for (; c>0; c--)
        {
                 for (b=199;b>0;b--)
                 {
                          for(a=1;a>0;a--);
                 }      
        }
            
}
/*******************************************************************************
* 函 数 名         : LcdWriteCom
* 函数功能                   : 向LCD写入一个字节的命令
* 输    入         : com
* 输    出         : 无
*******************************************************************************/
#ifndef         LCD1602_4PINS         //当没有定义这个LCD1602_4PINS时,执行八位数据线,写入程序
void LcdWriteCom(uchar com)          //写入命令
{
        LCD1602_E = 0;     //使能
        LCD1602_RS = 0;           //选择发送命令
        LCD1602_RW = 0;           //选择写入
        
        LCD1602_DATAPINS = com;     //放入命令
        Lcd1602_Delay1ms(1);                //等待数据稳定
        LCD1602_E = 1;                  //写入时序
        Lcd1602_Delay1ms(5);          //保持时间
        LCD1602_E = 0;
}
#else //有定义这个LCD1602_4PINS时,执行以下四位数据线写入程序,以下未执行
void LcdWriteCom(uchar com)          //写入命令
{
        LCD1602_E = 0;         //使能清零
        LCD1602_RS = 0;         //选择写入命令
        LCD1602_RW = 0;         //选择写入
        LCD1602_DATAPINS = com;        //由于4位的接线是接到P0口的高四位,所以传送高四位不用改
        Lcd1602_Delay1ms(1);
        LCD1602_E = 1;         //写入时序
        Lcd1602_Delay1ms(5);
        LCD1602_E = 0;
        LCD1602_DATAPINS = com << 4; //发送低四位
        Lcd1602_Delay1ms(1);
        LCD1602_E = 1;         //写入时序
        Lcd1602_Delay1ms(5);
        LCD1602_E = 0;
}
#endif
/*******************************************************************************
* 函 数 名         : LcdWriteData
* 函数功能                   : 向LCD写入一个字节的数据
* 输    入         : dat
* 输    出         : 无
*******************************************************************************/                  
#ifndef         LCD1602_4PINS                   //四位数据,开发板用的八位数据
void LcdWriteData(uchar dat)                        //写入数据
{
        LCD1602_E = 0;        //使能清零
        LCD1602_RS = 1;        //选择输入数据
        LCD1602_RW = 0;        //选择写入
        LCD1602_DATAPINS = dat; //写入数据
        Lcd1602_Delay1ms(1);
        LCD1602_E = 1;   //写入时序
        Lcd1602_Delay1ms(5);   //保持时间
        LCD1602_E = 0;
}
#else
void LcdWriteData(uchar dat)                        //写入数据 8位数据 以下未执行
{
        LCD1602_E = 0;          //使能清零
        LCD1602_RS = 1;          //选择写入数据
        LCD1602_RW = 0;          //选择写入
        LCD1602_DATAPINS = dat;        //由于4位的接线是接到P0口的高四位,所以传送高四位不用改
        Lcd1602_Delay1ms(1);
        LCD1602_E = 1;          //写入时序
        Lcd1602_Delay1ms(5);
        LCD1602_E = 0;
        LCD1602_DATAPINS = dat << 4; //写入低四位
        Lcd1602_Delay1ms(1);
        LCD1602_E = 1;          //写入时序
        Lcd1602_Delay1ms(5);
        LCD1602_E = 0;
}
#endif
/*******************************************************************************
* 函 数 名       : LcdInit()
* 函数功能                 : 初始化LCD屏
* 输    入       : 无
* 输    出       : 无
*******************************************************************************/                  
#ifndef                LCD1602_4PINS
void LcdInit()                                                  //LCD初始化子程序 开发板是八位数据
{
         LcdWriteCom(0x38);  //功能设置指令 0011 1000,采用几位数据线
        LcdWriteCom(0x0c);  //开显示不显示光标 0000 1100
        LcdWriteCom(0x06);  //写一个指针加1 0000 0110
        LcdWriteCom(0x01);  //清屏  0000 0001
        //LcdWriteCom(0x80);  //设置数据指针起点 1000 0000,显示位置起点默认增加00H+80H表示第一行。40H+80H=C0H表示第二行
}
#else
void LcdInit()                                                  //LCD初始化子程序四位数据线,开发板是八位数据,以下未执行
{
        LcdWriteCom(0x32);         //将8位总线转为4位总线
        LcdWriteCom(0x28);         //在四位线下的初始化
        LcdWriteCom(0x0c);  //开显示不显示光标
        LcdWriteCom(0x06);  //写一个指针加1
        LcdWriteCom(0x01);  //清屏
        LcdWriteCom(0x80);  //设置数据指针起点
}
#endif
//程序结束
LCD1602不带转接板 8位 -温度和时间.zip (74.96 KB )

回帖(8)

申换换

2019-8-13 07:27:00
问题终于解决了,虽然不知道为什么,现在总算实现了同时显示温度和时间了。深层次原因只有待以后分析,目前发现不了。
关键点是写地址读数据函数,多添加几条指令,加粗红色字体代码,如下:
uint8 read_1302(uint8 addr)//先向1302写地址,再读数据
{
uint8  dat;
RST=0;
_nop_();
CLCK=0;
_nop_();
RST=1;
_nop_();
DS1302WriteByte(addr);//写地址
dat=DS1302RradByte();//读数据
CLCK=1;
_nop_();
CLCK=0;
_nop_();
RST=0;
_nop_();
  RST=1;
_nop_();
  _nop_();
IO = 0;
_nop_();
IO = 1;
_nop_();
return dat;
}
举报

周彦楠

2019-8-13 07:40:10
帮顶
举报

陈秀春

2019-8-13 07:48:34
感谢原子哥
举报

王学超

2019-8-13 08:05:00
哥哥,地址写的对吗?
举报

更多回帖

发帖
×
20
完善资料,
赚取积分