DB18B20恒温LCD1602显示(仿真真机测试过)30-40度

DB18B20有字一面 管脚是123, 1接gnd 2接51的一个i/0口, 3接5v电压。

LCD1602的接线图

恒温proteus仿真图,LCD1602首先要接入上拉电阻,电压太低初始化是不成功的。proteus4.8版本以上有DB18B20,蜂鸣器和继电器需要搭配三极管S8050 NPN管进行,不然电压不够无法驱动。温度大概控制在30-40度。在我收到任务时候对温度精确度不高,只要能控制在一定温度区间就可以。

 
#include <reg52.H>
#include <intrins.H>
#include <math.H>
 
#define uchar unsigned char
  #define uint unsigned int

 sbit rw = P2^1; //LCD1602读/写选择端
 sbit RS = P2^2;//LCD1602数据/命令选择端    
 sbit LCDEN = P2^0; //LCD1602使能端
	
 sbit BEEP=P3^7 ; //蜂鸣器驱动线
	sbit RELAY = P1^1;//继电器
	sbit LED=P2^4;
	sbit ds = P1^0;
	 void delayMs();
void delayUs()
{
    _nop_();  //空指令,延时一个指令周期
}

 void delayMs(uint a)
{
    uint i, j;
    for(i = a; i > 0; i--)
        for(j = 100; j > 0; j--);
 }
//继电器控制
  void relay_control(int val)
 {
    if(val==1)     
    {
	
		   		RELAY=1; //继电器吸合
		}
		else   //按下相应的按键
    {
			
		   		RELAY=0; //继电器释放
		  	
		}
 }
void writeComm(uchar comm) //写指令
{
	rw =0;
    RS = 0;    
    P0 = comm;
	  delayMs(1);
    LCDEN = 1;
    delayMs(1);
    LCDEN = 0;
    delayMs(1);
}
 
//写数据:RS=1, RW=0;
void writeData(uchar dat)
{
	
    RS = 1;
	rw =0;
    P0 = dat;
	delayMs(1);
    LCDEN = 1;
    delayMs(1);
    LCDEN = 0;
    delayMs(1);
 }
 
 
 void init() //LCD1602初始化函数
 {
    rw = 0; //LCD1602读/写选择端
   
    writeComm(0x38);//LCD1602显示设置
    writeComm(0x0c);//开显示 
    writeComm(0x06);//当读或写一个字符后地址指针加一,且光标加一
     writeComm(0x01);//清屏指令 
	 
}
 
void writeString(uchar * str, uchar length)//LCD1602写字符串
{
     uchar i;
    for(i = 0; i < length; i++)
    {
         writeData(str[i]);
     }
 }
 
/**//*****************************DS18B20*******************************/
 
/*温馨提示:读者若要彻底理解并会灵活使用DS18B20温度传感器,请详细阅读
中文手册,并且对照代码注释充分分析代码。如能静心分析,定能深有体会,获
益匪浅!*/

void dsInit()//初始化,详见中文手册
 {
    
    unsigned int i;  
    ds = 0;
    i = 100;  
    while(i>0) i--;    //用于延时(严格按照时序来写)
    ds = 1;   
    i = 4;
    while(i>0) i--;
 }
 
void dsWait()//等待信号
 {
      unsigned int i;
      while(ds);  
      while(~ds);
      i = 4;
      while(i > 0) i--;
}
 
 
bit readBit()//读一个bit位,一节数据共8个bit位,故后面会连续读到8次来实现读一个字节的数据
{
    unsigned int i;
    bit b;
    ds = 0;
    i++;   
    ds = 1; 
    i++; i++;  
    b = ds;
    i = 8; 
    while(i>0) i--;
    return b;//返回数据
}
 
unsigned char readByte() //读一个字节的数据
{
    unsigned int i;
    unsigned char j, dat;
    dat = 0;
    for(i=0; i<8; i++)//每一次读一位,一节数据共8位,读8次
    {
        j = readBit();
      
        dat = (j << 7) | (dat >> 1);
    }
    return dat;
}
 
 
void writeByte(unsigned char dat)//写一节数据
{
    unsigned int i;
    unsigned char j;
    bit b;
    for(j = 0; j < 8; j++)//每次写一位,写8次
    {
        b = dat & 0x01;//取最低位,从最低位泄气
        dat >>= 1; //右移一位   
        if(b)   
        {
           ds = 0;         
           i++; i++;//延时以产生时序  
           ds = 1;    
           i = 8;
           while(i>0) i--; //延时以产生时序 
        }
        else  
        {
          ds = 0;
          i = 8;
          while(i>0) i--;//延时以产生时序  
          ds = 1;
          i++; i++;//延时以产生时序
        }
   }
}
 
 
void sendChangeCmd()//发送指令
{
    dsInit();    
    //dsWait();   //lcd1602不能初始化关键这里
    delayMs(1);    
    writeByte(0xcc);//发送跳跃ROM指令(注意:当只有一只从机在总线上时,无论如何,
        //忽略ROM指令之后只能跟着发出一条暂存器指令【BEh】)
    writeByte(0x44);//发送温度转换指令
}
 
void sendReadCmd()//读操作
{
    dsInit();
    dsWait();
    delayMs(1);
    writeByte(0xcc);//发送跳跃ROM指令 
    writeByte(0xbe);//发送温度转换指令 
}
 
 
int getTmpValue()  //获取温度值
{
    unsigned int tmpvalue;
    int value; 
    float t;
    unsigned char low, high;
    sendReadCmd();
    
    low = readByte(); 
    high = readByte();
   
    tmpvalue = high;
    tmpvalue <<= 8;
    tmpvalue |= low;
    value = tmpvalue;
    
    t = value * 0.0625;
   
    value = t * 100 + (value > 0 ? 0.5 : -0.5); //大于0加0.5, 小于0减0.5
	  if(t>39)
		{
		 LED=1;
		 BEEP=~BEEP;
			relay_control(0);
		}
		else if(t<30)
		{
		 LED=1;
			relay_control(1);
		BEEP=1;
		}
		else
		{
			BEEP=0;
		 LED=0;
			relay_control(1);
		}
    return value;
}
 
void display(int v) //LCD1602显示内容设置
{
    unsigned char count;
    unsigned char datas[] = {0, 0, 0, 0, 0};
    unsigned int tmp = abs(v);
    datas[0] = tmp / 10000;
    datas[1] = tmp % 10000 / 1000;
    datas[2] = tmp % 1000 / 100;
    datas[3] = tmp % 100 / 10;
    datas[4] = tmp % 10;
    //writeComm(0xc0+3);
		writeComm(0x80+7);
    if(v < 0)
    {
			  
        writeString("- ", 2);
   }
    else
    {
       writeString("+ ", 2);
    }
    if(datas[0] != 0)
    {
        writeData('0'+datas[0]);
    }
    for(count = 1; count != 5; count++)
    {
        writeData('0'+datas[count]);
        if(count == 2)
        {
            writeData('.');
        }
    }
}
/**//*****************************DS18B20*******************************/
 
void main()
{
    uchar table[] = "TEMP: ";
	  LED=0;
	  BEEP=0;
	   RELAY=0;
	init();
    sendChangeCmd();
    
    writeComm(0x80);//这个写命令是设置你写的数据的起始位置,0x80是第一行第一位
    writeString(table, 6);//发送"  xianzaiwendu: "
    while(1)
    {
        delayMs(1000); //温度转换时间需要750ms以上
        //writeComm(0xc0);		
        display(getTmpValue());//通过LCD1602显示温度转换值
        sendChangeCmd();//发送转换指令		
    }
}

其中这段:void sendChangeCmd()//发送指令
{
    dsInit();    
    //dsWait();   //lcd1602不能初始化关键这里
    delayMs(1);    
    writeByte(0xcc);//发送跳跃ROM指令(注意:当只有一只从机在总线上时,无论如何,
        //忽略ROM指令之后只能跟着发出一条暂存器指令【BEh】)
    writeByte(0x44);//发送温度转换指令
}

是很多码友在实际真机测试时候无法实现的关键。我猜是延迟不对,超时引起的。dsWait()注释后,真机就可以显示了。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值