DS18B20温度传感器——测试环境温度及代码

醒醒!,还在睡呢,开始干代码了!

1.简介:

单片机通过OneWire协议与DS18B20通信,最终测出环境温度

OneWire 总线的硬件接口很简单,只需要把 DS18B20 的数据引脚和单片机的一个 IO 口接上

2.DS18B20引脚及说明

 

 

 说明:GND接地,DQ单引线用于数据的输入,VDD接电源正极(注意正负极不能接反)

3.储存形式

通过编程,将二进制的数字来表示温度

 可以看出最高五位为符号位,若全为0则温度为正值,全为1则为负值

最后四位为小数位,最高可以精确到2的-4次方,可以知道每次变化最小变化值位0.0625

 例如+25.0625°C是

 1*2的-4次方+1*2的0次方+1*2的三次方+1*2的4次方

由于高五位都是0 所以最终结果是+25.0625°C;

二.各个模块的原理

2.1.初始化

在初始化阶段,总线控制器先拉低总线至少480微秒(这里可以拉低500微秒),然后释放总线

进入接收状态,等待 15-60us,然后发出一个由 60-240us低电平信号构成的存在脉冲。

unsigned char OneWire_Init()
{
	unsigned char i;
	unsigned char AckBit;
	OneWire_DQ=1;
	OneWire_DQ=0;
	i = 247;while (--i);		//Delay 500us
	OneWire_DQ=1;
	i = 32;while (--i);			//Delay 70us
	AckBit=OneWire_DQ;
	i = 247;while (--i);		//Delay 500us
	return AckBit;
}

先拉低,延时500微秒等待,再拉高,延时70微秒等待,然后把应答位取出来,再延时500微秒返回应答位,0位应答,1为非应答

这里延时多少可以去用软件来生成代码

2.2.写时序

有两种写时序:写 1 时序和写 0 时序,总线控制器通过写 1 时序写逻辑 1 到DS18B20,写 0 时序写逻辑 0 到 DS18B20,所有写时序必须最少持续 60us

1.发送一位

void OneWire_SendBit(unsigned char Bit)
{
	unsigned char i;
	OneWire_DQ=0;
	i = 4;while (--i);			//Delay 10us
	OneWire_DQ=Bit;
	i = 24;while (--i);			//Delay 50us
	OneWire_DQ=1;
}

 先把总线拉低,延迟10微秒,将bit赋值给DQ(把要发送的数据即bit放在总线上,发0即保持低电平,发1即变成高电平),然后延时50us,最后释放总线;

这里十分巧妙,给的数据是0的话就是0,不拉高总线,是1的话会拉高,延时50us后看变化如果是低电平说明发送的是0,高电平发送1,最后把总线拉高,这里10+50刚好等于60 满足最小的60us

                             2.发送一个字节(一字节也就是八位)

将上述情况重复八次就好

void OneWire_SendByte(unsigned char Byte)
{
	unsigned char i;
	for(i=0;i<8;i++)
	{
		OneWire_SendBit(Byte&(0x01<<i));
	}
}

从低到高依次将Byte的每一位取出来。(一般取数据用&,将某个数据置1用 | )

2.3.读时序

总线控制器发起读时序时,DS18B20 仅被用来传输数据给控制器。因此,总线控制器在发出读暂存器指令[BEh]或读电源模式指令[B4H]后必须立刻开始读时序。

2.3.1接收(读)一位:

主机将总线拉低1~15us,然后释放总线,并在拉低后15us内读取总线电平(尽量贴近15us末尾,这样成功的概率高) 读取为低电平则为

接收0,读取为高电平则为接收1,整个时间片应大于60us

 

unsigned char OneWire_ReceiveBit()
{
	unsigned char i;
	unsigned char Bit;
	OneWire_DQ=0;
	i = 2;while (--i);			//Delay 5us
	OneWire_DQ=1;
	i = 2;while (--i);			//Delay 5us
	Bit=OneWire_DQ;
	i = 24;while (--i);			//Delay 50us
	return Bit;
}

先拉低总线5us,再释放总线5us(此时控制权交给从机),现在时序到了10us,如果读取的数据是0,那么从机就强制拉低总线,如果读取到的数据是1,那就维持高电平,再延时50us满足时序需求.

2.3.2接收(读)一个字节:

和发送一个字节一样

将接收一位重复八次

unsigned char OneWire_ReceiveByte()
{
	unsigned char i;
	unsigned char Byte=0x00;
	for(i=0;i<8;i++)
	{
		if(OneWire_ReceiveBit()){Byte|=(0x01<<i);}
	}
	return Byte;
}

这里定义一个Byte返回值初值为0x00,然后用Byte按位或0x01(从低位到高位)就可以得到Byte了.

上述的发送一位,一个字节;接收一位,一个字节,可以封装到一个文件中,起名为OneWire

下面是OneWire.c

#include <REGX52.H>


sbit OneWire_DQ=P3^7;


unsigned char OneWire_Init()
{
	unsigned char i;
	unsigned char AckBit;
	OneWire_DQ=1;
	OneWire_DQ=0;
	i = 247;while (--i);		//Delay 500us
	OneWire_DQ=1;
	i = 32;while (--i);			//Delay 70us
	AckBit=OneWire_DQ;
	i = 247;while (--i);		//Delay 500us
	return AckBit;
}


void OneWire_SendBit(unsigned char Bit)
{
	unsigned char i;
	OneWire_DQ=0;
	i = 4;while (--i);			//Delay 10us
	OneWire_DQ=Bit;
	i = 24;while (--i);			//Delay 50us
	OneWire_DQ=1;
}


unsigned char OneWire_ReceiveBit()
{
	unsigned char i;
	unsigned char Bit;
	OneWire_DQ=0;
	i = 2;while (--i);			//Delay 5us
	OneWire_DQ=1;
	i = 2;while (--i);			//Delay 5us
	Bit=OneWire_DQ;
	i = 24;while (--i);			//Delay 50us
	return Bit;
}

void OneWire_SendByte(unsigned char Byte)
{
	unsigned char i;
	for(i=0;i<8;i++)
	{
		OneWire_SendBit(Byte&(0x01<<i));
	}
}


unsigned char OneWire_ReceiveByte()
{
	unsigned char i;
	unsigned char Byte=0x00;
	for(i=0;i<8;i++)
	{
		if(OneWire_ReceiveBit()){Byte|=(0x01<<i);}
	}
	return Byte;
}

下面是OneWire.h

#ifndef __ONEWIRE_H__
#define __ONEWIRE_H__
unsigned char OneWire_Init();
void OneWire_SendBit(unsigned char Bit);
unsigned char OneWire_ReceiveBit();
void OneWire_SendByte(unsigned char Byte);
unsigned char OneWire_ReceiveByte();


#endif

接下来是DS18B20的

ConverT():发送指令,跳过ROM,温度变换

然后是读取温度,在读取函数里面发送完两条指令后,控制权交给从机

这里数据处理比较难理解

剖析:TLSB为第八位,TMSB为高八位

Temp=(TMSB<<8)|TLSB;//将高八位左移八位,把数据整合成16位的;

最后还要把Temp/16

为什么要除以16呢?

这里还是看上面存储形式的那张图片,最低四位为小数位,但是我们整合出来的Temp最低四位是整数位,那么我们只需将Temp右移动4位即可,在二进制里面向右移动一位相当于 /2,那么向右移四位就是 /16了。

#include <REGX52.H>
#include "OneWire.h"


#define DS18B20_SKIP_ROM			0xCC
#define DS18B20_CONVERT_T			0x44
#define DS18B20_READ_SCRATCHPAD 	0xBE
void DS18B20_ConvertT()
{
	OneWire_Init();
	OneWire_SendByte(DS18B20_SKIP_ROM);
	OneWire_SendByte(DS18B20_CONVERT_T);
}


float DS18B20_ReadT()
{
	unsigned char TLSB,TMSB;
	int Temp;
	float T;
	OneWire_Init();
	OneWire_SendByte(DS18B20_SKIP_ROM);
	OneWire_SendByte(DS18B20_READ_SCRATCHPAD);
	TLSB=OneWire_ReceiveByte();
	TMSB=OneWire_ReceiveByte();
	Temp=(TMSB<<8)|TLSB;
	T=Temp/16.0;
	return T;
}
#ifndef __DS18B02_H__
#define __DS18B02_H__

void DS18B20_ConvertT();
float DS18B20_ReadT();


#endif

最后是主函数测试,这里要用到LCD1602显示屏

#include <REGX52.H>
#include "LCD1602.h"
#include "DS18B02.h"
#include "Delay.h"

float T;

void main()
{
	DS18B20_ConvertT();		
	Delay(1000);			
	LCD_Init();
	LCD_ShowString(1,1,"Temperature:");
	while(1)
	{
		DS18B20_ConvertT();	
		T=DS18B20_ReadT();	
		if(T<0)				
		{
			LCD_ShowChar(2,1,'-');	
			T=-T;			
		}
		else				
		{
			LCD_ShowChar(2,1,'+');	
		}
		LCD_ShowNum(2,2,T,3);		
		LCD_ShowChar(2,5,'.');		
		LCD_ShowNum(2,6,(unsigned long)(T*10000)%10000,4);
	}
}

逻辑全在代码里面了,容易体会,要注意最后

         LCD_ShowNum(2,2,T,3);        
        LCD_ShowChar(2,5,'.');        
        LCD_ShowNum(2,6,(unsigned long)(T*10000)%10000,4);//这里

温度要想显示出四位小数无误,要将float强转为long类型的,但是又不能直接转

所以我们先将整数部分取出显示出来,然后将T*10000%10000就可以取出小数部分了

这里就是DS18B20的基本内容了,还可以实现报警等等功能,在此基础上稍作修改即可,

精心摸鱼(hhh)

 

  • 5
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
#include<reg52.h> #define ui unsigned int #define uc unsigned char //宏定义 sbit SET=P3^1; //定义调整键 sbit DEC=P3^2; //定义减少键 sbit ADD=P3^3; //定义增加键 sbit BEEP=P3^6; //定义蜂鸣器 sbit ALAM=P1^2; //定义灯光报警 sbit ALAM1=P1^4; sbit DQ =P3^7; //定义DS18B20总线I/O sbit SCL=P1^6; sbit SDA=P1^7; sbit DIAN=P0^5; //小数点 bit bdata shanshuo_st; //闪烁间隔标志 bit bdata beep_st; //蜂鸣器间隔标志 uc x=0; //计数器 ui bai,shi,ge; uc set_st=0; //状态标志 char shangxian,xiaxian; code LEDData[]={0x5F,0x44,0x9D,0xD5,0xC6,0xD3,0xDB,0x47,0xDF,0xD7,0xCF,0xDA,0x9B,0xDC,0x9B,0x8B}; //====================================DS18B20========================================= /*****延时子程序*****/ void Delay_DS18B20(int num) { while(num--) ; } void delay()//5微秒延时函数 { ;; } void start() //开始信号 { SDA=1; delay(); SCL=1; delay(); SDA=0; delay(); } void stop() //终止信号 { SDA=0; delay(); SCL=1; delay(); SDA=1; delay(); } void respons() //应答 { uc i; SCL=1; delay(); while((SDA==1)&&(i<250))i++;//如果SDA为低应答有效,或者超过一定时间默认应答有效 SCL=0; delay(); } void init24c04()//I2C总线初始化 { SDA=1; delay(); SCL=1; delay(); } void write_byte(uc date)//写操作 { uc i,temp; temp=date; for(i=0;i<8;i++) { temp=temp<<1; SCL=0; delay(); SDA=CY; delay(); SCL=1; delay(); } SCL=0; delay(); SDA=1; delay(); } uc read_byte()//读操作 { uc i,k; SCL=0; delay(); SDA=1; delay(); for(i=0;i<8;i++) { SCL=1; delay(); k=(k<<1)|SDA; SCL=0; delay(); } return k; } void write_add(uc address,uc date)//往任意地址存数据 { start(); write_byte(0xa0);//0xa0代表写入 respons(); write_byte(address); respons(); write_byte(date); respons(); stop(); } uc read_add(uc address)//读随意地址内容 { uc date; start(); write_byte(0xa0);//0xa0代表写入 respons(); write_byte(address); respons(); start(); write_byte(0xa1);//0xa1代表读出 respons(); date=read_byte(); stop(); return date;//返回值 } /*****初始化DS18B20*****/ void Init_DS18B20(void) { uc w=0; DQ = 1; //DQ复位 Delay_DS18B20(8); //稍做延时 DQ = 0; //单片机将DQ拉低 Delay_DS18B20(80); //精确延时,大于480us DQ = 1; //拉高总线 Delay_DS18B20(14); W=DQ //稍做延时后,如果x=0则初始化成功,x=1则初始化失败 Delay_DS18B20(20); } /*****读一个字节*****/ unsigned char ReadOneChar(void) { uc i=0; uc dat = 0; for (i=8;i>0;i--) { DQ = 0; // 给脉冲信号 dat>>=1; DQ = 1; // 给脉冲信号 if(DQ) dat|=0x80; Delay_DS18B20(4); } return(dat); } /*****写一个字节*****/ void WriteOneChar(uc dat) { uc i=0; for (i=8; i>0; i--) { DQ = 0; DQ = dat&0x01; Delay_DS18B20(5); DQ = 1; dat>>=1; } } /*****读取温度*****/ ui ReadTemperature(void) { ui b=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); } /*****延时子程序*****/ void Delay(ui num) { while( --num ); } /*****初始化定时器0*****/ void InitTimer(void) { TMOD=0x01; TH0=0x3c; TL0=0xb0; //50ms(晶振12M) } /*****读取温度*****/ void check_wendu(void) { ui f; f=ReadTemperature()-5; //获取温度值并减去DS18B20的温漂误差 if(f<0)f=0; if(f>999)f=999; bai=f/100; //计算得到十位数字 shi=(f0)/10; //计算得到个位数字 ge=(f0); //计算得到整数位 } /*****显示开机初始化等待画面*****/ void Disp_init(void) { P0= ~0x80; //显示---- P2= 0x7F; Delay(200); P2=0XDF; Delay(200); P2 = 0xF7; Delay(200); P2= 0xFD; Delay(200); P2= 0xFF; //关闭显示 } /*****显示温度子程序*****/ void Disp_Temperature(void) //显示温度 { P0= ~0x98; //显示C P2= 0x7F; Delay(400); P0=LEDData[ge]; //显示个位 P2 = 0xDF; Delay(400); P0 =LEDData[shi]; //显示十位 DIAN = 0; //显示小数点 P2= 0xF7; Delay(400); P0 =~LEDData[bai]; //显示百位 P2 = 0xFD; Delay(400); P2 = 0xff; //关闭显示 } /*****显示报警温度子程序*****/ void Disp_alarm(uc baojing) { p0 =~0x98; //显示C p2 = 0x7F; Delay(200); p0 =~LEDData[baojing]; //显示十位 P0 =~LEDData[baojing/10]; //显示百位 P2 = 0xF7; Delay(200); if(set_st==1)P0 =~0xCE; else if(set_st==2)P0 =~0x1A; //上限H、下限L标示 P2= 0xFD; Delay(200); P2 = 0xff; //关闭显示 } /*****报警子程序*****/ void Alarm() { if(x>=10){beep_st=~beep_st;x=0;} if((bai*10+shi)>=shangxian&&beep;_st==1) { BEEP=0; ALAM1=0; } else if((bai*10+shi)>=shangxian&&beep;_st==0) { BEEP=1; ALAM1=0; } if((bai*10+shi)<xiaxian&&beep;_st==1) { BEEP=0; ALAM=0; } else if((bai*10+shi)<xiaxian&&beep;_st==0) { BEEP=1; ALAM=0; } if(((bai*10+shi)<shangxian)&&((bai*10+shi)>=xiaxian)) { BEEP=1; ALAM1=1; ALAM=1; } } /*****主函数*****/ void main(void) { ui z; InitTimer(); //初始化定时器 EA=1; //全局中断开关 TR0=1; ET0=1; //开启定时器0 check_wendu(); check_wendu(); shangxian=read_add(10); xiaxian=read_add(20); for(z=0;z<300;z++) { Disp_init(); } while(1) { if(SET==0) { Delay(2000); do{} while(SET==0); set_st++;x=0;shanshuo_st=1; if(set_st>2)set_st=0; } if(set_st==0) { check_wendu(); Disp_Temperature(); Alarm(); //报警检测 } else if(set_st==1) { BEEP=1; //关闭蜂鸣器 ALAM=1; ALAM1=1; if(x>=10){shanshuo_st=~shanshuo_st;x=0;} if(shanshuo_st) {Disp_alarm(shangxian);} if(ADD==0) { do{Disp_alarm(shangxian);} shangxian++; if(shangxian>99)shangxian=99; write_add(10,shangxian); } else if(DEC==0) { do{Disp_alarm(shangxian);} while(DEC==0); shangxian--; if(shangxian<xiaxian) shangxian=xiaxian; write_add(10,shangxian); } } else if(set_st==2) { BEEP=1; //关闭蜂鸣器 ALAM=1; ALAM1=1; if(x>=10){shanshuo_st=~shanshuo_st;x=0;} if(shanshuo_st) {Disp_alarm(xiaxian);} if(ADD==0) { do{Disp_alarm(xiaxian);} while(ADD==0); xiaxian++; if(xiaxian>shangxian) xiaxian=shangxian; write_add(20,xiaxian); } else if(DEC==0) { do{Disp_alarm(xiaxian);} while(DEC==0); xiaxian--; if(xiaxian<0) xiaxian=0; write_add(20,xiaxian); } } } } /*****定时器0中断服务程序*****/ void timer0(void) interrupt 1 { TH0=0x3c; TL0=0xb0; x++; }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值