微波EDA网,见证研发工程师的成长!
首页 > 研发问答 > 嵌入式设计讨论 > MCU和单片机设计讨论 > 请大家看看我的 18B20程序为什么不能在液晶屏上显示啊谢谢

请大家看看我的 18B20程序为什么不能在液晶屏上显示啊谢谢

时间:10-02 整理:3721RD 点击:
                                                                                                                                                                                          
#include <STC12C5A60S2.H> //单片机头文件
#include <intrins.h>        //51基本运算(包括_nop_空函数)
#include <float.h>     //浮点数处理
#include <string.h>
/*****************************************************************************/
//温度传感器定义
sbit DQ = P2 ^ 5;//ds18B20
/********************************************************************************************/
typedef unsigned char      uint8;          // 无符号8位整型变量 //
/********************************************************************************************
/
/********************************************************************************************/
#define                LCM2402_DB0_DB7                P0                        // 定义LCM2402的数据总线
sbit LCM2402_RS   = P2 ^ 5;                                        // 定义LCM2402的RS控制线
sbit LCM2402_RW   = P2 ^ 6;                                        // 定义LCM2402的RW控制线
sbit LCM2402_E    = P2 ^ 7;                                        // 定义LCM2402的E控制线
sbit LCM2402_Busy = P0 ^ 7;                                        // 定义LCM2402的测忙线(与LCM2402_DB0_DB7关联)
data unsigned char TIME_DD,TIME_MO,TIME_YY,TIME_WW,TIME_HH,TIME_MM,TIME_SS;//设置日、月、年、周、时、分、秒和温度存放区
data bit DAY_BIT = 0;//天数增加标志位(用于日期进位的启动)
data unsigned char DIS_BIT = 0; //多种信息的切换显示
data unsigned char cou  = 0;     // 软计数器,对10ms时基信号累加到1s
/********************************************************************************************
/********************************************************************************************/
#define                        CMD_clear                0x01             // 清除屏幕
#define                        CMD_back                0x02             // DDRAM回零位
#define                        CMD_dec1                0x04             // 读入后AC(指针)减1,向左写
#define                        CMD_add1                0x06             // 读入后AC(指针)加1,向右写
#define                        CMD_dis_gb1                0x0f             // 开显示_开光标_开光标闪烁
#define                        CMD_dis_gb2                0x0e             // 开显示_开光标_关光标闪烁
#define                        CMD_dis_gb3                0x0c             // 开显示_关光标_关光标闪烁
#define                        CMD_OFF_dis                0x08             // 关显示_关光标_关光标闪烁
#define                        CMD_set82                0x38             // 8位总线_2行显示
#define                        CMD_set81                0x30             // 8位总线_1行显示(上边行)
#define                        CMD_set42                0x28             // 4位总线_2行显示
#define                        CMD_set41                0x20             // 4位总线_1行显示(上边行)
#define                        lin_1                        0x80             // 4位总线_1行显示(上边行)
#define                        lin_2                        0xc0             // 4位总线_1行显示(上边行)
//DS18B20温度传感器驱动(显示0至60度)
/******************************************************************************/
void Delay(int num){//延时函数
        while(num--) ;
}


/******************************************************************************/
void Init_DS18B20(void){//初始化ds1820
        unsigned char x=0;
        DQ = 1;    //DQ复位
        Delay(8);  //稍做延时
        DQ = 0;    //单片机将DQ拉低
        Delay(80); //精确延时 大于 480us
        DQ = 1;    //拉高总线
        Delay(14);
        x=DQ;      //稍做延时后 如果x=0则初始化成功 x=1则初始化失败
        Delay(20);
}
/******************************************************************************/
unsigned char ReadOneChar(void){//读一个字节
        unsigned char i=0;
        unsigned char dat = 0;
        for (i=8;i>0;i--){
                DQ = 0; // 给脉冲信号
                dat>>=1;
                DQ = 1; // 给脉冲信号
                if(DQ==1)
                dat|=0x80;
                Delay(4);
        }
        return(dat);
}
/******************************************************************************/
void WriteOneChar(unsigned char dat){//写一个字节
        unsigned char i=0;
        for (i=8; i>0; i--){
                DQ = 0;
                DQ = dat&0x01;
                Delay(5);
                DQ = 1;
                dat>>=1;
        }
}
/******************************************************************************/
unsigned int ReadTemperature(void){//读取温度
        unsigned char a=0;
        unsigned char b=0;
        unsigned int t=0;
        float tt=0;
        Init_DS18B20();
        WriteOneChar(0xCC); // 跳过读序号列号的操作
        WriteOneChar(0x44); // 启动温度转换
        Init_DS18B20();
        WriteOneChar(0xCC); //跳过读序号列号的操作
        WriteOneChar(0xBE); //读取温度寄存器
        a=ReadOneChar();  //读低8位
        b=ReadOneChar(); //读高8位
        t=b;
        t<<=8;
        t=t|a;
        tt=t*0.0625;
        t= tt*10+0.5; //放大10倍输出并四舍五入
        return(t);
}        /*****************************************************************************/
                       
/*********************************************************************************************
函数名:毫秒级CPU延时函数
调  用:DELAY_MS (?);
参  数:1~65535(参数不可为0)
返回值:无
结  果:占用CPU方式延时与参数数值相同的毫秒时间
备  注:应用于1T单片机时i<600,应用于12T单片机时i<125
/*********************************************************************************************/
void DELAY_MS (unsigned int a){
        unsigned int i;
        while( --a != 0){
                for(i = 0; i < 600; i++);
        }
}
/*********************************************************************************************/
/********************************************************************************************
// 读LCM忙程序 [底层协议] // (所有底层协议都无需关注)
// LCM2402测忙,若LCM2402处于忙状态,本函数将等待至非忙状态 //
/********************************************************************************************/
void LCM2402_TestBusy(void){
           LCM2402_DB0_DB7 = 0xff;                //设备读状态
           LCM2402_RS = 0;
           LCM2402_RW = 1;
           LCM2402_E = 1;
           while(LCM2402_Busy);                //等待LCM不忙
           LCM2402_E = 0;                                //
}
/********************************************************************************************
// 写指令程序 //
// 向LCM2402写命令 本函数需要1个指令集的入口参数 //
/********************************************************************************************/
void LCM2402_WriteCMD(uint8 LCM2402_command) {
          LCM2402_TestBusy();
          LCM2402_RS = 0;
          LCM2402_RW = 0;
          LCM2402_DB0_DB7 = (LCM2402_command/16)<<4;//0x45 0x40         
          LCM2402_E = 1;
          LCM2402_E = 0;
          LCM2402_DB0_DB7 = (LCM2402_command%16)<<4;//0x45 0x50
          LCM2402_E = 1;
          LCM2402_E = 0;
}
/********************************************************************************************
// 写数据程序 //
// 向LCM2402写数据 //
/********************************************************************************************/
void LCM2402_WriteData(uint8 LCM2402_data){
    LCM2402_TestBusy();
          LCM2402_RS = 1;
          LCM2402_RW = 0;
        LCM2402_DB0_DB7 = (LCM2402_data/16)<<4;
          LCM2402_E = 1;
          LCM2402_E = 0;
        LCM2402_DB0_DB7 = (LCM2402_data%16)<<4;
          LCM2402_E = 1;
          LCM2402_E = 0;
}
/********************************************************************************************
// 打印字符串程序 // (本函数调用指针函数)
// 向LCM发送一个字符串,长度48字符之内
// 第一行位置 0x00~0x17  第二行位置 0x40~0x57
// 应用举例:print(0x80,"net"); //在第一行第一位处从左向右打印\net字符串
/********************************************************************************************/
void print(uint8 a,uint8 *str){
        LCM2402_WriteCMD(a | 0x80);
        while(*str != '\0'){
                LCM2402_WriteData(*str++);
        }
        *str = 0;
}
/********************************************************************************************
// 打印单字符程序 //
// 第一行位置 0x00~0x17  第二行位置 0x40~0x57
// 向LCM发送一个字符,以十六进制(0x00)表示
// 应用举例:print(0xc0,0x30); //在第二行第一位处打印字符“0”
/********************************************************************************************/
void print2(uint8 a,uint8 t){
                LCM2402_WriteCMD(a | 0x80);
                LCM2402_WriteData(t);
}
/********************************************************************************************
// 定义小汉字 //
// 可写入8个自字义字符,写入后可用其CGRAM代码直接提取显示。
// 字符定义方法请参考技术手册
/********************************************************************************************/
uint8 code Xword[]={
    0x18,0x18,0x07,0x08,0x08,0x08,0x07,0x00,        //℃,代码 0x00
    0x00,0x00,0x00,0x00,0xff,0x00,0x00,0x00,        //一,代码 0x01
    0x00,0x00,0x00,0x0e,0x00,0xff,0x00,0x00,        //二,代码 0x02
    0x00,0x00,0xff,0x00,0x0e,0x00,0xff,0x00,        //三,代码 0x03
    0x00,0x00,0xff,0xf5,0xfb,0xf1,0xff,0x00,        //四,代码 0x04
    0x00,0xfe,0x08,0xfe,0x0a,0x0a,0xff,0x00,        //五,代码 0x05
    0x00,0x04,0x00,0xff,0x00,0x0a,0x11,0x00,        //六,代码 0x06
    0x00,0x1f,0x11,0x1f,0x11,0x11,0x1f,0x00,        //日,代码 0x07
};
void CgramWrite(void) {        // 装入CGRAM //
    uint8 i;
        LCM2402_WriteCMD(0x06);                        // CGRAM地址自动加1
        LCM2402_WriteCMD(0x40);                        // CGRAM地址设为00处
    for(i=0;i<64;i++) {
            LCM2402_WriteData(Xword[i]);// 按数组写入数据
    }
}
/********************************************************************************************
// LCM2402初始化 //(使用者可自定义,加 * 号程序行必须保留但可修改)
/********************************************************************************************/
void LCM2402_Init(void){
          LCM2402_WriteCMD(CMD_set42);        //* 显示模式设置:显示2行,每个字符为5*7个像素
          LCM2402_WriteCMD(CMD_set42);        //* 显示模式设置:显示2行,每个字符为5*7个像素
          LCM2402_WriteCMD(CMD_clear);        //  显示清屏
          LCM2402_WriteCMD(CMD_back);                //* 数据指针指向第1行第1个字符位置
          LCM2402_WriteCMD(CMD_add1);                //  显示光标移动设置:文字不动,光标右移
          LCM2402_WriteCMD(CMD_dis_gb3);         //  显示开及光标设置:显示开,光标开,闪烁开
        CgramWrite();                                        //  向CGRAM写入自定义字符
}
/********************************************************************************************/
//                        以上是LCM2402驱动程序                        //
/*********************************************************************************************/

/**********************************************************************************************/       
/********************************************************************************************
/*********************************************************************************************
函数名:10位A/D转换初始化函数
调  用:Read_init (?);
参  数:输入的端口(0000 0XXX 其中XXX是设置输入端口号,可用十进制0~7表示,0表示P1.0,7表示P1.7)
返回值:无
结  果:开启ADC功能并设置ADC的输入端口
备  注:使用ADC功能时需要将对应的IO接口设置为高阻输入方式(例如:P1M1 = 0x01;)
/**********************************************************************************************/
void Read_init (unsigned char CHA){
        unsigned char AD_FIN=0; //存储A/D转换标志
    CHA &= 0x07;            //选择ADC的8个接口中的一个(0000 0111 清0高5位)
    ADC_CONTR = 0x40;                //ADC转换的速度(0XX0 0000 其中XX控制速度,请根据数据手册设置)
    _nop_();
    ADC_CONTR |= CHA;       //选择A/D当前通道
    _nop_();
    ADC_CONTR |= 0x80;      //启动A/D电源
    DELAY_MS(1);            //使输入电压达到稳定(1ms即可)
}
//显示温度
void RealTime_Display(void){
  unsigned int t;
         
                  t=ReadTemperature();
           // print(0x80,"20");
            print2(0x4A,t/10+0x30);
                Delay(5);
            print2(0x4B,t%10+0x30);
                Delay(5);
                print2(0x4C,0x00);            // 显示c
                Delay(5);
       
}
/**********************************************************************************************/
/*********************************************************************************************
函数名:10位A/D转换函数
调  用:? = ADC_Read();
参  数:无
返回值:10位ADC数据高(从0到1023(十进制))
结  果:读出指定ADC接口的A/D转换值,并返回数值
备  注:适用于STC12C5A60S2系列单片机(必须使用STC12C5A60S2.h头文件)
/**********************************************************************************************/
unsigned int ADC_Read (void){
        unsigned char AD_FIN=0; //存储A/D转换标志
    ADC_CONTR |= 0x08;      //启动A/D转换(0000 1000 令ADCS = 1)
    _nop_();
    _nop_();
    _nop_();
    _nop_();
    while (AD_FIN ==0){     //等待A/D转换结束
        AD_FIN = (ADC_CONTR & 0x10); //0001 0000测试A/D转换结束否
    }
    ADC_CONTR &= 0xE7;      //1111 0111 清ADC_FLAG位, 关闭A/D转换,
return (ADC_RES*4+ADC_RESL);//返回A/D转换结果(10位ADC数据高8位在ADC_RES中,低2位在ADC_RESL中)
}
/**********************************************************************************************/
/**********************************************************************************************/
//ADC显示程序
void ADCshow ()
{unsigned int m;
unsigned int n;
Read_init(0);
m = ADC_Read ();
print(0x80,"shandong");
print2(0x45,m/1000+0x30);//
print2(0x46,(m%1000)/100+0x30);//
print2(0x47,(m%100)/10+0x30);//
print2(0x48,m%10+0x30);//
  DELAY_MS(5);
Read_init(1);
n = ADC_Read ();                           
print(0x80,"shandong");
print2(0x40,n/1000+0x30);//
print2(0x41,(n%1000)/100+0x30);//
print2(0x42,(n%100)/10+0x30);//
print2(0x43,n%10+0x30);//  
}
/********************************************************************************************/
// 测试用函数 //
void main (void){
        Init_DS18B20();                        
        LCM2402_Init();//LCM2402初始化
         DELAY_MS(5);
        ReadTemperature();
        P1M1 = 0x03;//
//        P1M0 = 0x00;                           
//        Read_init (1);  
        while(1){ //主线程//
        RealTime_Display( );
         DELAY_MS(5);
         ADCshow();
         
                }
        }

是连定值都不显示,还是显示一个值,但是不改变,而且很大或者很小

这个不仅程序长,而且错误究竟是啥也没说清楚 啊

最近在学 感觉好难

好有压力!

怎么感觉很麻烦的样子,自己写的相对简单一点!

可以了,延时短了

这个 能不恩给你告诉我们究竟是哪里有问题啊

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

网站地图

Top