DS18B20温度传感器触发继电器

有关代码:

DS18B20

//temp.h

#ifndef __TEMP_H_
#define __TEMP_H_

#include<reg52.h>
//---重定义关键词---//
#ifndef uchar
#define uchar unsigned char
#endif

#ifndef uint 
#define uint unsigned int
#endif

//--定义使用的IO口--//
sbit DSPORT=P3^7;

//--声明全局函数--//
void Delay1ms(uint );
uchar Ds18b20Init();
void Ds18b20WriteByte(uchar com);
uchar Ds18b20ReadByte();
void  Ds18b20ChangTemp();
void  Ds18b20ReadTempCom();
int Ds18b20ReadTemp();

#endif
//temp.c

#include "temp.h"

void Delay1ms(uint y)
{
	uint x;
	for( ; y>0; y--)
	{
		for(x=110; x>0; x--);
	}
}

//初始化,成功返回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;          //初始化成功
}

//向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;
	}
}

//读取一个字节
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;
}

//转换温度
void  Ds18b20ChangTemp()
{
	Ds18b20Init();
	Delay1ms(1);
	Ds18b20WriteByte(0xcc);		//跳过ROM操作命令		 
	Ds18b20WriteByte(0x44);	    //温度转换命令
	//Delay1ms(100);	        //等待转换成功,而如果你是一直刷着的话,就不用这个延时了
   
}

//发送读取温度命令
void  Ds18b20ReadTempCom()
{	

	Ds18b20Init();
	Delay1ms(1);
	Ds18b20WriteByte(0xcc);	 //跳过ROM操作命令
	Ds18b20WriteByte(0xbe);	 //发送读取温度命令
}

//读取温度
int Ds18b20ReadTemp()
{
	int temp = 0;
	uchar tmh, tml;
	Ds18b20ChangTemp();			 	//先写入转换命令
	Ds18b20ReadTempCom();			//然后等待转换完后发送读取温度命令
	tml = Ds18b20ReadByte();		//读取温度值共16位,先读低字节
	tmh = Ds18b20ReadByte();		//再读高字节
	temp = tmh;
	temp <<= 8;
	temp |= tml;
	return temp;
}

LCD

//lcd.h

#ifndef __LCD_H_
#define __LCD_H_

//#define LCD1602_4PINS

#include<reg52.h>

//---重定义关键词---//
#ifndef uchar
#define uchar unsigned char
#endif

#ifndef uint 
#define uint unsigned int
#endif

//接线
#define LCD1602_DATAPINS P0
sbit LCD1602_E=P2^7;
sbit LCD1602_RW=P2^5;
sbit LCD1602_RS=P2^6;

void Lcd1602_Delay1ms(uint c);   //误差 0us
/*LCD1602写入8位命令子函数*/
void LcdWriteCom(uchar com);
/*LCD1602写入8位数据子函数*/	
void LcdWriteData(uchar dat);
/*LCD1602初始化子程序*/		
void LcdInit();	
void LcdSetCursor(uchar x,uchar y);  //坐标显示
void LcdShowStr(uchar x,uchar y,uchar *str);     //显示字符串
void DisplayOneChar(uchar X, uchar Y, uchar DData);	  

#endif
//lcd.c

#include "lcd.h"

void Lcd1602_Delay1ms(uint c)   //延时1ms,误差 0us,该函数是在12MHZ晶振下,12分频单片机的延时。
{
    uchar a,b;
		for (; c>0; c--)
		{
			 for (b=199;b>0;b--)
			 {
					for(a=1;a>0;a--);
			 }      
		}
    	
}

//向LCD写入一个字节的命令
#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 
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

//向LCD写入一个字节的数据		
#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(10);     //保持时间
		LCD1602_E = 0;
}
#else
void LcdWriteData(uchar dat)     //写入数据
{
		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

//初始化LCD屏	   
#ifndef		LCD1602_4PINS
void LcdInit()//LCD初始化子程序
{
		LcdWriteCom(0x38);        //开显示
		LcdWriteCom(0x0c);        //开显示不显示光标
		LcdWriteCom(0x06);        //写一个指针加1
		LcdWriteCom(0x01);        //清屏
		LcdWriteCom(0x80);        //设置数据指针起点
}
#else
void LcdInit()//LCD初始化子程序
{
		LcdWriteCom(0x32);        //将8位总线转为4位总线
		LcdWriteCom(0x28);        //在四位线下的初始化
		LcdWriteCom(0x0c);        //开显示不显示光标
		LcdWriteCom(0x06);        //写一个指针加1
		LcdWriteCom(0x01);        //清屏
		LcdWriteCom(0x80);        //设置数据指针起点
}
#endif

//坐标显示
void LcdSetCursor(uchar x,uchar y)
{
    uchar addr;
    if(y == 0)
        addr = 0x00 + x;
    else
        addr = 0x40 + x;
    
    LcdWriteCom(addr|0x80);
}

//按指定位置显示一个字符
void DisplayOneChar(uchar X, uchar Y, uchar DData)
{
		Y &= 0x1;
		X &= 0xF;            //限制X不能大于15,Y不能大于1
		if (Y) X |= 0x40;    //当要显示第二行时地址码+0x40;
		X |= 0x80;           //算出指令码
		LcdWriteCom(X);      //发命令字
		LcdWriteData(DData); //发数据
}
					
//显示字符串
void LcdShowStr(uchar x,uchar y,uchar *str)
{
    LcdSetCursor(x,y);       //当前字符的坐标
    while(*str != '\0')
    {
        LcdWriteData(*str++);
    }
}

main.c

/*接线
* P37-->J14
* P14-->J15
*/

#include "reg52.h"
#include "intrins.h"
#include "math.h"
#include "stdio.h"
#include "temp.h"	
#include "lcd.h"

typedef unsigned int u16;	  //对数据类型进行声明定义
typedef unsigned char u8;

sbit relay=P1^4;	   
u16 getTemp;

char num=0;
u8 show[]="TMP";

void delay(u16 i)
{
	while(i--);	
}

void datapros(int temp) 	 
{
  float tp;  
	if(temp< 0)				//当温度值为负数
  {
		DisplayOneChar(2,1,'-'); 	  //   -
		temp=temp-1;
		temp=~temp;
  }
 	else
  {			
		DisplayOneChar(2,1,' ');
	}
	tp=temp;
	temp=tp*0.0625*100+0.5;	
	/*留两个小数点就*100,+0.5是四舍五入,因为C语言浮点数转换为整型的时候把小数点
	* 后面的数自动去掉,不管是否大于0.5,而+0.5之后大于0.5的就是进1了,小于0.5的就
	* 算加上0.5,还是在小数点后面。
    */
	LcdShowStr(0,0,show);
	DisplayOneChar(3,0,temp/1000+'0');
	DisplayOneChar(4,0,temp%1000/100+'0');
	DisplayOneChar(5,0,'.');	
	DisplayOneChar(6,0,temp%1000%100/10+'0');
	DisplayOneChar(7,0,temp%1000%10%10+'0');
	getTemp=temp%1000;
}

void main()
{	
	u16 y=0;
	LcdInit();
	while(1)
	{
		datapros(Ds18b20ReadTemp());	 //数据处理函数
		delay(500);
		if(y==0)
		{
			if(getTemp<20)             //低于20°C触发
			{
				y=1;
			}
		}
		if(y==1)
		{
			relay=0;
		}
	}		
}

效果:

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值