DS18B20温度传感器使用介绍

本文详细介绍了DS18B20数字温度传感器的工作原理、特性、引脚功能、操作流程及其实现温度测量的代码示例。这种传感器通过1-Wire接口通信,支持总线结构,具有宽温度范围和抗干扰能力。文章还展示了如何使用单片机读取和转换温度数据,并提供了温度数据的存储格式。
摘要由CSDN通过智能技术生成

DS18B20温度传感器

DS18B20介绍

DS18B20是一种常见的数字温度传感器,其控制命令和数据都是以数字信号的方式输入输出,相比较于模拟温度传感器,具有功能强大、硬件简单、易扩展、抗干扰性强等特点
测温范围:-55°C 到 +125°C
通信接口:1-Wire(单总线)
其它特征:可形成总线结构、内置温度报警功能、可寄生供电VCC都不用接,直接信号线加GND线就可以通信

模拟温度传感器:热敏电阻,可通过温度的变化改变电阻值,一般再接一个分压电阻,串联到VCC和GND之间,需要用AD转换芯片将模拟信号转换为数字信号才能供单片机使用

DS18B20内部集成了模拟温度传感器所需的电路,其内部也相当于有个小芯片,将模拟信号处理成数字信号后存到RAM中,再通过引脚,将信号传给单片机使用

引脚及应用电路

引脚功能
VDD电源(3.0V ~ 5.5V)
GND电源地
DQ单总线接口

在这里插入图片描述

内部结构框图

在这里插入图片描述

  1. 如果使用寄生供电,则需要外加强上拉驱动
  2. VDD电源检测,当没有接VCC时,会调整内部状态,使用寄生供电方式
  3. 64-BIT ROM:作为器件地址,用于总线通信的寻址
    SCRATCHPAD(暂存器):用于总线的数据交互
    EEPROM:用于保存温度触发阈值和配置参数

存储器结构

在这里插入图片描述

DS18B20操作流程

初始化:从机复位,主机判断从机是否响应
ROM操作:ROM指令+本指令需要的读写操作
功能操作:功能指令+本指令需要的读写操作

ROM指令功能指令
SEARCH ROM [F0h](搜索ROM)CONVERT T [44h](温度变换,启动温度传感器读取温度放到RAM中)
READ ROM [33h](读ROM)WRITE SCRATCHPAD [4Eh](写RAM,加某个地址后,把数据写入到RAM中)
MATCH ROM [55h](匹配ROM,发送该指令后会紧接着发送ROM地址,用于多个设备)READ SCRATCHPAD [BEh](读RAM,可以只读取RAM前两位的温度值,后续的不读取)
SKIP ROM [CCh](跳过ROM,只有一个设备时使用)COPY SCRATCHPAD [48h](复制暂存器,调用该指令后会将RAM Byte2,3,4的值写入到EEPROM中,掉电不丢失)
ALARM SEARCH [ECh](报警ROM)RECALL E2 [B8h](将EEPROM的值读取到RAM对应地址处)
READ POWER SUPPLY [B4h](判断是否是寄生供电)

DS18B20数据帧

温度变换:初始化→跳过ROM →开始温度变换

在这里插入图片描述

#include "OneWire.h"		//该头文件在单总线部分

#define SKIP_ROM 			0xCC
#define CONVERT_T 			0x44
#define READ_SCRATCHPAD 	 0xBE
/**
  * @brief DS18B20温度转换,即将温度传感器的值读入到RAM中
  * @param无
  * @retval无
  */
void DS18B20_ConvertT(void)
{
	OneWire_Init();
	OneWire_SendByte(SKIP_ROM);		//单总线发送一个字节地址
	OneWire_SendByte(CONVERT_T);
}

温度读取:初始化→跳过ROM →读暂存器→连续的读操作

在这里插入图片描述

/**
  * @brief读取温度供main函数使用
  * @param无
  * @retval返回浮点型的温度值
  */
float DS18B20_ReadT()
{
	unsigned char TLSB,TMSB;
	unsigned int temp;
	float T;
	OneWire_Init();
	OneWire_SendByte(SKIP_ROM);
	OneWire_SendByte(READ_SCRATCHPAD);
	TLSB = OneWire_ReceiveByte();		//接收温度低位
	TMSB = OneWire_ReceiveByte();		//接收温度高位
	temp = (TMSB<<8)|TLSB;		//组成一个16位的数值
	
	/*因为LS BYTE的最低位应该是BIT4,BIT3~0都是小数位,而temp的值是把小数位当成了整数了
	导致整个有效数值向左移动了4位,要将4位小数也显示出来,temp就要除以16.0,因为不是除以
	16,所以小数位也是保留了下来*/
	
	T = temp /16.0;
	return T;
}

温度存储格式

在这里插入图片描述

在这里插入图片描述

温度数据都是以补码的形式存储的,读取时要进行转换,特别是负数

实例所用的知识点:

1.动态扫描按键,一直按着按键不放,DS18B20依然能检测温度并显示

2.最高值和最低值存储在E2PROM中,断电后仍能读取

main函数中使用

#include <REGX52.H>
#include "DS18B20.h"
#include "LCD1602.h"
unsigned char receive;
float temperature;

void main()
{
	LCD_Init();
	LCD_ShowString(1,1,"Temperature");
	DS18B20_ConvertT();				//先进行温度转换
	Delay1ms(1000);					//延时1秒,避免下面读取温度时会闪一下出现默认的温度值
	while(1)
	{
         DS18B20_ConvertT();		//在循环中不断进行温度转换,将温度值读到DS18B20的RAM中
		temperature = DS18B20_ReadT();		//从RAM中读读取温度值
		if(temperature < 0)				   //如果温度是负数
		{
			LCD_ShowChar(2,1,'-');
			temperature = -temperature;		//转为正数
		}
		else
		{
			LCD_ShowChar(2,1,'+');
		}
		LCD_ShowNum(2,2,temperature,3);
		LCD_ShowChar(2,5,'.');
		/*因为要在LCD1602上显示小数,但没有显示小数的函数,
		且小数也不能通过求余得出,所以要将数值乘以10000,
		强转为unsigned long,再求余10000,得出小数部分,
		整数和小数分开显示(如果想输出两位小数,则*100后%)*/
		LCD_ShowNum(2,6,(unsigned long)(temperature*10000)%10000,4);
	}
}
#include #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<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(f999)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)=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;z2)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=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++; }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值