51单片机温度传感器DS18B20


51单片机温度传感器DS18B20

实现功能

插上DS18B20温度传感器,数码管显示检测的温度值

单片机型号:STC89C52

DS18B20介绍

1、DS18B20简介

DS18B20 是由 DALLAS 半导体公司推出的一种的“一线总线(单总线)”接口的温度传感器。与传统的热敏电阻等测温元件相比,它是一种新型的体积小、 适用电压宽、与微处理器接口简单的数字化温度传感器。

2、DS18B20内部结构

图片[1] - 51单片机温度传感器DS18B20 - MaxSSL

DS18B20 温度传感器的内部存储器包括一个高速的暂存器 RAM 和一个非易 失性的可电擦除的EEPROM,后者存放高温度和低温度触发器 TH、TL 和配置寄存器。 配置寄存器是配置不同的位数来确定温度和数字的转化,配置寄存器结构如下:

图片[2] - 51单片机温度传感器DS18B20 - MaxSSL

低五位一直都是”1″,TM 是测试模式位,用于设置 DS18B20 在工作模式还 是在测试模式。在 DS18B20 出厂时该位被设置为 0,用户不需要去改动。R1 和 R0 用来设置 DS18B20 的精度(分辨率),可设置为 9,10,11 或 12 位,对应的分辨率温度是 0.5℃,0.25℃,0.125℃和 0.0625℃。R0 和 R1 配置如下 图:

图片[3] - 51单片机温度传感器DS18B20 - MaxSSL

当温度转换命令(44H)发布后,经转换所得的温度值以二字节补码形式存放在高速暂存存储器的第 0 和第 1 个字节。存储的两个字节,高字节的前 5 位是符号位 S,单片机可通过单线接口读到该数据,读取时低位在前,高位在后, 数据格式如下:

图片[4] - 51单片机温度传感器DS18B20 - MaxSSL

如果测得的温度大于 0,这 5 位为‘ 0’,只要将测到的数值乘以 0.0625 (默认精度是 12 位)即可得到实际温度;如果温度小于 0,这 5 位为‘ 1’, 测到的数值需要取反加 1 再乘以 0.0625 即可得到实际温度。温度与数据对应关系如下:

图片[5] - 51单片机温度传感器DS18B20 - MaxSSL

比如我们要计算+85 度,数据输出十六进制是 0X0550,因为高字节的高 5 位为 0,表明检测的温度是正温度,0X0550 对应的十进制为 1360,将这个值乘以12 位精度 0.0625,所以可以得到+85 度。

3、信号时序

由于 DS18B20 是单总线器件,所有的单总线器件都要求采用严格的信号时序,以保证 数据的 完整性。DS18B20 时序包括如下几种:初始化时序、写(0 和 1)时序、 读(0 和 1)时序。 DS18B20 发送所有的命令和数据都是字节的低位在前。这里我们 简单介绍这几个信号的时序:

(1)初始化时序

单总线上的所有通信都是以初始化序列开始。主机输出低电平,保持低电平时间至少 480us(该时间的时间范围可以从 480 到 960 微妙),以产生复位脉冲。接着主机释放总线,外部的上拉电阻将单总线拉高,延时 15~60 us,并进入接收模式。接着 DS18B20 拉低总线 60~240 us,以产生低电平应答脉冲,若为低电平,还要做延时,其延时的时间从外部上拉电阻将单总线拉高算起最少要480 微妙。初始化时序图如下:

图片[6] - 51单片机温度传感器DS18B20 - MaxSSL

(2)写时序

写时序包括写 0 时序和写 1 时序。所有写时序至少需要 60us,且在 2 次独立的写时序之间至少需要 1us 的恢复时间,两种写时序均起始于主机拉低总线。写 1 时序:主机输出低电平,延时 2us,然后释放总线,延时 60us。写 0 时序:主机输出低电平,延时 60us,然后释放总线,延时 2us。写时序图如下:

图片[7] - 51单片机温度传感器DS18B20 - MaxSSL

(3)读时序

单总线器件仅在主机发出读时序时,才向主机传输数据,所以,在主机发出读数据命令后,必须马上产生读时序,以便从机能够传输数据。所有读时序至少需要 60us,且在 2 次独立的读时序之间至少需要 1us 的恢复时间。每个读时序都由主机发起,至少拉低总线 1us。主机在读时序期间必须释放总线,并且在时序起始后的 15us 之内采样总线状态。读时序图如下:

图片[8] - 51单片机温度传感器DS18B20 - MaxSSL

典型的读时序过程为:主机输出低电平延时 2us,然后主机转入输入模式延 时 12us,然后读取单总线当前的电平,然后延时 50us。 296 在了解了单总线时序之后,我们来看看 DS18B20 的典型温度读取过程, DS18B20 的典型温度读取过程为:复位→发 SKIP ROM 命令(0XCC)→发开始转 换命令(0X44)→延时→复位→发送 SKIP ROM 命令(0XCC)→发读存储器命令 (0XBE)→连续读出两个字节数据(即温度)→结束

硬件原理

图片[9] - 51单片机温度传感器DS18B20 - MaxSSL

从上图可以看出,传感器接口的单总线管脚接至单片机 P3.7 IO 口上,在介绍单总线的时候我们说过,为了让单总线默认为高电平,通常会在单总线上接上拉电阻,在图中并没有看到有上拉电阻,这是因为单片机 IO 都外接了 10K 上拉 电阻,当单片机 IO 口连接到传感器的总线管脚时即相当于它们外接上拉电阻, 所以此处可以省去。

软件编写

程序框架如下:

  • 编写数码管显示功能
  • 编写 DS18B20 读取温度功能
  • 编写主函数
#include #include //数码管管脚定义#define LEDP0    //定义数码管位选信号控制脚sbit LSA=P2^2;sbit LSB=P2^3;sbit LSC=P2^4;//DS18B20管脚定义sbit DS18B20_PORT=P3^7;//共阴极数码管显示0~F的段码数据unsigned char Smg[17]={0x3f,0x06,0x5b,0x4f,0x66,0x6d,0x7d,0x07,0x7f,0x6f,0x77,0x7c,0x39,0x5e,0x79,0x71};//延迟函数void Delay(unsigned char x){while(x--);}//数码管显示函数void Smg_Display(unsigned char Data[],unsigned char Location){unsigned char i=0;unsigned char temp=Location-1;for(i=temp;i<8;i++){   switch(i)  //位选{case 0: LSC=1;LSB=1;LSA=1;break;case 1: LSC=1;LSB=1;LSA=0;break;case 2: LSC=1;LSB=0;LSA=1;break;case 3: LSC=1;LSB=0;LSA=0;break;case 4: LSC=0;LSB=1;LSA=1;break;case 5: LSC=0;LSB=1;LSA=0;break;case 6: LSC=0;LSB=0;LSA=1;break;case 7: LSC=0;LSB=0;LSA=0;break;}LED=Data[i-temp];   //传送段选数据Delay(100);         //延时一段时间,等待显示稳定LED=0x00;           //消影}}//初始化时序unsigned char DS18B20_Init(){//复位DS18B20DS18B20_PORT=0;   //拉低DQDelay(75);        //拉低 750usDS18B20_PORT=1;   //拉高DQDelay(2);         //拉高 20us//检测DS18B20是否存在unsigned char time=0;while((DS18B20_PORT==1)&&(time<20))  //等待 DQ 为低电平{   time++;Delay(1);}if(time>=20)  //如超时则强制返回return 0;elsetime=0;while((DS18B20_PORT==0)&&(time<20))  //等待 DQ 为高电平{time++;Delay(1);}if(time>=20)   //如超时则强制返回return 0;else return 1;}//写时序void DS18B20_Write_Byte(unsigned char Data){unsigned char i;unsigned char temp;for(i=0;i<8;i++)      //循环8次,每次写一位,且先写低位再写高位{temp=Data&0x01;   //选择低位准备写入Data>>=1;         //将次高位移到低位if(temp==1)       //写 1 时序{   DS18B20_PORT=0;_nop_();      //延时 1us_nop_();DS18B20_PORT=1;Delay(6);}else              //写 0 时序{DS18B20_PORT=0;Delay(6);DS18B20_PORT=1;_nop_();_nop_();}}}//读时序unsigned char DS18B20_Read_Byte(){unsigned char i;unsigned char dat=0;unsigned char Data=0;for(i=0;i<8;i++)        //循环8次,每次读取一位,且先写低位再写高位{DS18B20_PORT=0;     //先拉低电平_nop_();_nop_();DS18B20_PORT=1;    //再释放总线_nop_();_nop_(); //该段时间不能过长,必须在 15us 内读取数据if(DS18B20_PORT==1) //如果总线上为 1 则数据为 1,否则为 0dat=1;elsedat=0;Delay(5);Data=(dat<<7)|(Data>>1);   //将读取的数字合在一起}return Data;}//获取温度值float DS18B20_Read_Temperture(){float temp;                 //存储温度值unsigned char DataH=0;      //存储低字节unsigned char DataL=0;      //存储高字节unsigned int value=0;DS18B20_Init();             //复位DS18B20_Write_Byte(0xcc);   //发 SKIP ROM 命令(0XCC)DS18B20_Write_Byte(0x44);   //发开始转换命令(0x44)                            //延时(可有可无)DS18B20_Init();             //复位DS18B20_Write_Byte(0xcc);   //发 SKIP ROM 命令(0XCC)DS18B20_Write_Byte(0xBE);   //发读存储器命令(0xBE)                            //连续读出两个字节数据(即温度)DataL=DS18B20_Read_Byte();  //低字节DataH=DS18B20_Read_Byte();  //高字节value=(DataH<<8)+DataL;     //合并为16位数据if((value&0xf800)==0xf800)  //判断符号位  负温度{value=(~value)+1;       //取反加一,因为是以反码形式存储temp=value*(-0.0625);   //乘以精度}else                        //正温度temp=value*0.0625;return temp;}void main(){int value;unsigned char i=0;unsigned char buf[5];DS18B20_Init();    //初始化 DS18B20while(1){i++;if(i%50==0)    //间隔一段时间读取温度值,间隔时间要大于温度传感器转换温度时间value=DS18B20_Read_Temperture()*10;   //保留温度值小数后一位if(value<0)    //负温度{value=-value;buf[0]=0x40;    //显示负号}elsebuf[0]=0x00;    //不显示buf[1]=Smg[value/1000];              //百位buf[2]=Smg[value%1000/100];          //十位buf[3]=Smg[value%1000%100/10]|0x80;  //个位+小数点buf[4]=Smg[value%1000%100%10];       //小数点后一位Smg_Display(buf,4);}}
© 版权声明
THE END
喜欢就支持一下吧
点赞0 分享