【常用传感器】DS18B20温度传感器原理详解及例程代码_ds18b20温度传感器工作原理

img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上物联网嵌入式知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新

需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)

如果你需要这些资料,可以戳这里获取

  • 返回16位二进制温度数值
  • 主机和从机通信使用单总线,即使用单线进行数据的发送和接收
  • 在使用中不需要任何外围元件,独立芯片即可完成工作。
  • 掉电保护功能 DS18B20 内部含有 EEPROM ,通过配置寄存器可以设定数字转换精度和报警温度,在系统掉电以后,它仍可保存分辨率及报警温度的设定值。
  • 每个DS18B20都有独立唯一的64位-ID,此特性决定了它可以将任意多的DS18b20挂载到一根总线上,通过ROM搜索读取相应DS18B20的温度值
  • 宽电压供电,电压2.5V~5.5V
  • DS18B20返回的16位二进制数代表此刻探测的温度值,其高五位代表正负。如果高五位全部为1,则代表返回的温度值为负值。如果高五位全部为0,则代表返回的温度值为正值。后面的11位数据代表温度的绝对值,将其转换为十进制数值之后,再乘以0.0625即可获得此时的温度值。
传感器引脚及原理图

DS18B20传感器的引脚及封装图如下:

在这里插入图片描述
DS18B20一共有三个引脚,分别是:

  • GND:电源地线
  • DQ:数字信号输入/输出端。
  • VDD:外接供电电源输入端。

单个DS18B20接线方式: VDD接到电源,DQ接单片机引脚,同时外加上拉电阻,GND接地

注意这个上拉电阻是必须的,就是DQ引脚必须要一个上拉电阻

DS18B20上拉电阻

首先我们要知道什么是漏极开路

首先看一下 场效应管(MOSFET)
在这里插入图片描述
场效应管 是电压控制型元器件,只要对栅极施加一定电压,DS就会导通。

漏极开路:MOS管的栅极G和输入连接,源极S接公共端,漏极D悬空(开路)什么也没有接,直接输出 ,这时只能输出低电平和高阻态,不能输出高电平

那么这个时候会出现三种情况:

  1. 图a为正常输出(内有上拉电阻):场效应管导通时,输出低电位输出低电位,截止时输出高电位。
  2. 图b为漏极开路输出,外接上拉电阻:场效应管导通时,驱动电流是从外部的VCC流经电阻通过MOSFET到GND,输出低电位,截止时输出高电位。
  3. 图c为漏极开路输出,无外接上拉电阻:场效应管导通时输出低电位,截止呈高阻态(断开)。
    在这里插入图片描述

总结:

开漏输出只能输出低电平,不能输出高电平。漏极开路输出高电平时必须在输出端与正电源(VCC)间外接一个上拉电阻。否则只能输出高阻态。

DS18B20 是单线通信,即接收和发送都是这个通信脚进行的。 其接收数据时为高电阻输入,其发送数据时是开漏输出,本身不具有输出高电平的能力,即输出0时通过MOS下拉为低电平,而输出1时,则为高阻,需要外接上拉电阻将其拉为高电平。 因此,需要外接上拉电阻,否则无法输出1。

外接上拉电阻阻值:

DS18B20的工作电流约为1mA,VCC一般为5V,则电阻R=5V/1mA=5KΩ,

所以正常选择4.7K电阻,或者相近的电阻值。

DS18B20寄生电源

DSl8B20的另一个特点是不需要再外部供电下即可工作。当总线高电平时能量由单线上拉电阻经过DQ引脚获得。高电平同时充电一个内部电容,当总线低电平时由此电容供应能量。这种供电方法被称为“寄生电源”。另外一种选择是DSl8B20由接在VDD的外部电源供电
在这里插入图片描述

DS18B20内部构成

主要由以下3部分组成: 64 位ROM,高速暂存器,存储器

64 位ROM存储独有的序列号

ROM中的64位序列号是出厂前被光刻好的,它可以看作是该DS18B20的地址序列码,每个DS18B20的64位序列号均不相同。这样就可以实现一根总线上挂接多个DS18B20的目的。

高速暂存器包含:

  • 温度传感器
  • 一个字节的温度上限和温度下限报警触发器(TH和TL)
  • 配置寄存器允许用户设定9位,10位,11位和12位的温度分辨率,分别对应着温度的分辨率为:0.5°C,0.25°C,0.125°C,0.0625°C,默认为12位分辨率,

存储器:由一个高速的RAM和一个可擦除的EEPROM组成,EEPROM存储高温和低温触发器(TH和TL)以及配置寄存器的值,(就是存储低温和高温报警值以及温度分辨率)
在这里插入图片描述

DS18B20高速缓存器

高速暂存器由9个字节组成

  • 字节0~1 是温度存储器,用来存储转换好的温度。第0个字节存储温度低8位,第一个字节存储温度高8位
  • 字节2~3 是用户用来设置最高报警和最低报警值(TH和TL)。
  • 字节4 是配置寄存器,用来配置转换精度,可以设置为9~12 位。
  • 字节5~7 保留位。芯片内部使用
  • 字节8 CRC校验位。是64位ROM中的前56位编码的校验码。由CRC发生器产生。
    在这里插入图片描述
DS18B20温度读取与计算

DS18B20采用16位补码的形式来存储温度数据温度是摄氏度。当温度转换命令发布后,经转换所得的温度值以二字节补码形式存放在高速暂存存储器的第0和第1个字节。

高字节的五个S为符号位,温度为正值时S=1,温度为负值时S=0

剩下的11位为温度数据位,对于12位分辨率,所有位全部有效,对于11位分辨率,位0(bit0)无定义,对于10位分辨率,位0和位1无定义,对于9位分辨率,位0,位1,和位2无定义

在这里插入图片描述

对应的温度计算:


当五个符号位S=0时,温度为正值,直接将后面的11位二进制转换为十进制,再乘以0.0625(12位分辨率),就可以得到温度值;

当五个符号位S=1时,温度为负值,先将后面的11位二进制补码变为原码(符号位不变,数值位取反后加1),再计算十进制值。再乘以0.0625(12位分辨率),就可以得到温度值;

例如:

+125℃的数字输出07D0(00000111 11010000)
 
转换成10进制是2000,对应摄氏度:0.0625x2000=125°C

-55℃的数字输出为 FC90。
  
首先取反,然后+1,转换成原码为:11111011 01101111
数值位转换成10进制是870,对应摄氏度:-0.0625x870=-55°C

在这里插入图片描述
代码:

unsigned int Templ,Temp2,Temperature;  //Templ低八位,Temp2高八位
unsigned char Minus Flag=0;  //负温度标志位

if(Tenp2&0xFC)//判断符号位是否为1
{
	Minus Flag=l; //负温度标志位置1
	Temperature=((Temp2<<8)|Temp1); //高八位第八位进行整合
	Temperature=((Temperature)+1); //讲补码转换为原码,求反,补1
	Temperature\*=0.0625;//求出十进制
}
else   //温度为正值
{
	Minus Flag=0;  //负温度标志位置0
	Temperature =((Temp2<<8) |Temp1)\*0.0625;
}

配置寄存器

在配置寄存器中,我们可以通过R0和R1设置DS18B20的转换分辨率,DS18B20在上电后默认R0=1和R1=1(12分辨率),寄存器中的第7位和第0位到4位保留给设备内部使用。
在这里插入图片描述

单总线系统

在每个DS18B20内部都有一个唯一的64位长的序列号,这个序列号值就存在DS18B20内部的ROM中。开始的8位是产品类型编码(DS18B20是28H),接着的48位是每个器件唯一的序号,最后的8位是CRC校验码。
在这里插入图片描述

一线总线系统使用单总线主控来控制一个或多个从机设备。每个DS18B20都有独立唯一的64位-ID,此特性决定了它可以将任意多的DS18b20挂载到一根总线上,通过ROM搜索读取相应DS18B20的温度值。

DS18B20工作步骤

DS18B20的工作步骤可以分为三步:

1.初始化DS18B20
2.执行ROM指令
3.执行DS18B20功能指令

其中第二步执行ROM指令,也就是访问每个DS18B20,搜索64位序列号,读取匹配的序列号值,然后匹配对应的DS18B20,如果我们仅仅使用单个DS18B20,可以直接跳过ROM指令。而跳过ROM指令的字节是0xCC。

1.初始化DS18B20

任何器件想要使用,首先就是需要初始化,对于DS18B20单总线设备,首先初始化单总线为高电平,然后总线开始也需要检测这条总线上是否存在DS18B20这个器件。如果这条总线上存在DS18B20,总线会根据时序要求返回一个低电平脉冲,如果不存在的话,也就不会返回脉冲,即总线保持为高电平。

初始化具体时序步骤如下:

  • 1.单片机拉低总线至少480us,产生复位脉冲,然后释放总线(拉高电平)。
  • 2.这时DS8B20检测到请求之后,会拉低信号,大约60~240us表示应答。
  • 3.DS8B20拉低电平的60~240us之间,单片机读取总线的电平,如果是低电平,那么表示初始化成功
  • 4.DS18B20拉低电平60~240us之后,会释放总线。
    在这里插入图片描述
    代码如下:
/\*\*\*\*\*初始化DS18B20\*\*\*\*\*/
unsigned int Init\_DS18B20(void)
{
  unsigned int x=0;
  DQ = 1;      //DQ复位
  delay(4);    //稍做延时
  DQ = 0;      //单片机将DQ拉低
  delay(60);   //精确延时,大于480us
  DQ = 1;      //拉高总线
  delay(8);
  x = DQ;      //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
  delay(4);
  return x;
}

2.写时序

总线控制器通过控制单总线高低电平持续时间从而把逻辑1或0写DS18B20中。每次只传输1位数据

单片机想要给DS18B20写入一个0时,需要将单片机引脚拉低,保持低电平时间要在60~120us之间,然后释放总线
单片机想要给DS18B20写入一个1时,需要将单片机引脚拉低,拉低时间需要大于1us,然后在15us内拉高总线.

在写时序起始后15μs到60μs期间,DS18B20处于采样单总线电平状态。如果在此期间总线为高电平,则向DS18B20写入1;如果总线为低电平,则向DSl8B20写入0。

注意:2次写周期之间至少间隔1us

在这里插入图片描述

/\*\*\*\*\*写一个字节\*\*\*\*\*/
void WriteOneChar(unsigned char dat)
{
  unsigned char i=0;
  for (i=8; i>0; i--)
  {
    DQ = 0;
    DQ = dat&0x01;  //与1按位与运算,dat最低位为1时DQ总线为1,dat最低位为0时DQ总线为0
	delay(4);
    DQ = 1;
    dat>>=1;
  }
  delay(4);
}


DS18B20写入的功能命令:

ROM指令:

采用多个DS18B20时,需要写ROM指令来控制总线上的某个DS18B20
如果是单个DS18B20,直接写跳过ROM指令0xCC即可
在这里插入图片描述

RAM指令,DS18B20的一些功能指令

常用的是:

温度转换 0x44

开启温度读取转换,读取好的温度会存储在高速暂存器的第0个和第一个字节中

读取温度 0xBE
读取高速暂存器存储的数据
在这里插入图片描述

读时序

读时隙由主机拉低总线电平至少1μs然后再释放总线,读取DS18B20发送过来的1或者0

DS18B20在检测到总线被拉低1微秒后,便开始送出数据,若是要送出0就把总线拉为低电平直到读周期结束。若要送出1则释放总线为高电平。

在这里插入图片描述
注意:所有读时隙必须至少需要60us,且在两次独立的时隙之间至少需要1ps的恢复时间

同时注意:主机只有在发送读暂存器命令(0xBE)或读电源类型命令(0xB4)后,立即生成读时隙指令,DS18B20才能向主机传送数据。 也就是先发读取指令,再发送读时隙

最后一点: 写时序注意是先写命令的低字节,比如写入跳过ROM指令0xCC(11001100),写的顺序是“零、零、壹、壹、零、零、壹、壹”,

读时序时是先读低字节,在读高字节,也就是先读取高速暂存器的第0个字节(温度的低8位),在读取高速暂存器的第1个字节(温度的高8位) 我们正常使用DS18B20读取温度读取两个温度字节即可

51例程
sbit DQ=P1^0;	//定义DS18b20的管脚

/\*\*\*\*\*延时子程序\*\*\*\*\*/
void delay(unsigned int t)
{
	for(;t>0;t--);
}


/\*\*\*\*\*初始化DS18B20\*\*\*\*\*/
unsigned char Init\_DS18B20(void)
{
  unsigned char x=0;
  DQ = 1;      //DQ复位
  delay(8);    //稍做延时
  DQ = 0;      //单片机将DQ拉低
  delay(80);   //精确延时,大于480us
  DQ = 1;      //拉高总线
  delay(8);
  x = DQ;      //稍做延时后,如果x=0则初始化成功,x=1则初始化失败
  delay(4);
  return x;
}

/\*\*\*\*\*读一个字节\*\*\*\*\*/
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)
    	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(4);
    DQ = 1;
    dat>>=1;
  }
  delay(4);
}

/\*\*\*\*\*读取温度\*\*\*\*\*/
int ReadTemperature(void)
{
  	unsigned char a=0;
  	unsigned char b=0;
  	unsigned int t=0;
  	t=Init\_DS18B20();
  	if(t) return Real_temp;
  	WriteOneChar(0xCC);  //跳过读序号列号的操作
  	WriteOneChar(0x44);  //启动温度转换
  	t=Init\_DS18B20();
  	if(t) return Real_temp;
  	WriteOneChar(0xCC);  //跳过读序号列号的操作
  	WriteOneChar(0xBE);  //读取温度寄存器
  	a=ReadOneChar();     //读低8位
  	b=ReadOneChar();     //读高8位
  	t=b;
  	t<<=8;
  	t=t|a;
  	if(t<=0||t>0x900) 
	return Real_temp;
	t=t\*0.625+0.5;
  	return(t);
}


STM32例程

DS18B20.C

#include "ds18b20.h"
#include "delay.h" 

//复位DS18B20
void DS18B20\_Rst(void)	   
{                 
	DS18B20\_IO\_OUT(); 	//SET PG11 OUTPUT
    DS18B20_DQ_OUT=0; 	//拉低DQ
    delay\_us(750);    	//拉低750us
    DS18B20_DQ_OUT=1; 	//DQ=1 
	delay\_us(15);     	//15US


![img](https://img-blog.csdnimg.cn/img_convert/29ba53c929ecee1be98a15361e9300bc.png)
![img](https://img-blog.csdnimg.cn/img_convert/d6f2a04e185a14c7a2e965106fbea77d.png)

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上物联网嵌入式知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新**

**需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)**

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618679757)**

1 OUTPUT
    DS18B20_DQ_OUT=0; 	//拉低DQ
    delay\_us(750);    	//拉低750us
    DS18B20_DQ_OUT=1; 	//DQ=1 
	delay\_us(15);     	//15US


[外链图片转存中...(img-SokAo4IY-1715798258558)]
[外链图片转存中...(img-FFKlGGeo-1715798258558)]

**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上物联网嵌入式知识点,真正体系化!**

**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、电子书籍、讲解视频,并且后续会持续更新**

**需要这些体系化资料的朋友,可以加我V获取:vip1024c (备注嵌入式)**

**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618679757)**

  • 27
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
以下是基于STM32的DS18B20温度传感器例程: 1. 首先,你需要连接DS18B20到STM32,可以参考DS18B20数据手册来连接。一般来说,你需要将DS18B20的VCC接到STM32的5V引脚,将GND接到GND引脚,将DQ(数据线)接到STM32的GPIO引脚。 2. 接下来,在你的STM32项目中,你需要包含以下库文件: ``` #include "stm32f10x.h" #include "delay.h" #include "ds18b20.h" ``` 其中,"delay.h"是提供延时函数的头文件,"ds18b20.h"是DS18B20的头文件。 3. 在主函数中,你需要初始化GPIO引脚和DS18B20: ``` int main(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); // 使能GPIOB时钟 GPIO_InitStructure.GPIO_Pin = GPIO_Pin_0; // DQ引脚 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; // 最大输出速度50MHz GPIO_Init(GPIOB, &GPIO_InitStructure); // 初始化GPIOB0 ds18b20_init(); // 初始化DS18B20 while(1) { // 在这里读取DS18B20的温度并进行其他处理 } } ``` 4. 接下来,在"ds18b20.h"头文件中,你需要定义以下常量和函数: ``` #define DS18B20_PORT GPIOB // DS18B20使用的GPIO端口 #define DS18B20_PIN GPIO_Pin_0 // DS18B20使用的GPIO引脚 void ds18b20_init(void); void ds18b20_write_byte(uint8_t byte); uint8_t ds18b20_read_byte(void); float ds18b20_get_temp(void); ``` 其中,"ds18b20_init"函数用于初始化DS18B20,"ds18b20_write_byte"和"ds18b20_read_byte"函数用于向DS18B20写入字节并从DS18B20读取字节,"ds18b20_get_temp"函数用于获取DS18B20的温度。 5. 最后,在"ds18b20.c"文件中,你需要实现以上函数: ``` #include "ds18b20.h" void ds18b20_init(void) { GPIO_InitTypeDef GPIO_InitStructure; RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE); // 使能GPIOB时钟 GPIO_InitStructure.GPIO_Pin = DS18B20_PIN; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; // 开漏输出 GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(DS18B20_PORT, &GPIO_InitStructure); // 初始化DS18B20引脚 GPIO_SetBits(DS18B20_PORT, DS18B20_PIN); // 初始化时,DQ引脚输出高电平 } void ds18b20_write_byte(uint8_t byte) { uint8_t i; for(i = 0; i < 8; i++) { GPIO_ResetBits(DS18B20_PORT, DS18B20_PIN); // 先将DQ引脚拉低 delay_us(2); // 延时2us if(byte & 0x01) // 根据byte的最低位来决定输出高电平还是低电平 GPIO_SetBits(DS18B20_PORT, DS18B20_PIN); delay_us(60); // 延时60us GPIO_SetBits(DS18B20_PORT, DS18B20_PIN); // 将DQ引脚拉高 delay_us(2); // 延时2us byte >>= 1; // 将byte右移一位,继续处理下一位 } } uint8_t ds18b20_read_byte(void) { uint8_t i, byte = 0; for(i = 0; i < 8; i++) { GPIO_ResetBits(DS18B20_PORT, DS18B20_PIN); // 先将DQ引脚拉低 delay_us(2); // 延时2us GPIO_SetBits(DS18B20_PORT, DS18B20_PIN); // 将DQ引脚拉高 delay_us(2); // 延时2us byte >>= 1; // 先将byte右移一位 if(GPIO_ReadInputDataBit(DS18B20_PORT, DS18B20_PIN)) // 判断DQ引脚是否为高电平 byte |= 0x80; // 如果是,则将byte最高位置为1 delay_us(60); // 延时60us } return byte; } float ds18b20_get_temp(void) { uint8_t temp_l, temp_h; float temp; ds18b20_write_byte(0xcc); // 跳过ROM操作 ds18b20_write_byte(0x44); // 启动温度转换 delay_ms(750); // 等待转换完成(DS18B20最长转换时间为750ms) ds18b20_write_byte(0xcc); // 跳过ROM操作 ds18b20_write_byte(0xbe); // 读取温度寄存器 temp_l = ds18b20_read_byte(); // 先读取温度低8位 temp_h = ds18b20_read_byte(); // 再读取温度高8位 temp = (float)(temp_h << 8 | temp_l) / 16; // 将温度高8位和低8位合并并转换成浮点数 return temp; } ``` 以上就是基于STM32的DS18B20温度传感器例程,希望对你有帮助!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值