当前位置: 首页 > news >正文

51单片机读取DS18B20温度传感器

1.首先我们知道DS18B20是单总线协议,只有一根数据线。所以Data数据线即使发送端又是接收端,同时DS18B20内部接了弱上拉电阻(如图一所示),数据线默认为高电平。有了这些概念,我们就能进行下一步。

                                               图一(截取DS18B20芯片手册)

2.初始化DS18B20

     看图二可知,首先我们将总线从高拉低,保持480us左右,然后释放总线(没人拉低,默认高电平),此时总线上为高电平。然后等待15-60us,等待DS18B20模块将总线拉低。(如果拉低则说明DS18B20响应成功),DS18B20拉低60-240us后,释放总线。初始化完成。

                                                              图二

 3.向DS18B20写入数据。

        3.1 向DS18B20写0。

       首先将总线由高拉低,低电平保持15us到60us,然后释放总线。写入0完成。

        3.2 向DS18B20写1。

        将总线由高拉低,低电平保持1-15us(看图三可知,低电平时间要低于15us),再释放总线。

 

                                                                 图三

4.读取DS18B20寄存器中数据

        4.1 读数据0

将总线由高拉低,保持1-15us。如果DS18B20中数据是0,则会将总线拉低。此时,总线上为低电平。我们要在60us以内读取数据(图四可知,60us以内DS18B20会释放总线)。否则可能会读到错误数据。

        4.2 读数据1

将总线由高拉低,保持1-15us,如果数据是1,DS18B20会释放总线,总线上为高电平。然后在15us以后读取数据。

   

                                                                    图四

5.DS18B20一些命令

  5.1 0xCC→跳过ROM  

 5.2 其他命令

  0x44→开始温度转换指令 ,0xBE→读取数据寄存器指令

                                                                     图五

6.DS18B20数据寄存器

 这里也比较关键,所以拎出来讲一下。

首先DS18B20的数据寄存器是16位,其中寄存器高字节的高5位的S表示符号位,其余3位才是表示数据。这里就浅讲一下,后面结合代码来解析一下。

                                                                     图六 

 7.代码实例

    声明:首先这里面的延时都是按照经验值,就是通过实践,得出比较准确的延时。

    注:这里使用的是12M频率的单片机,如果是1M可以把for循环去掉

延时函数如下:

  sbit DQ=P1^4;
//我这里是P1^4连接了DS18B20的数据线。
//你们根据自己的单片机原理图查询
void Delay_OneWire(unsigned int t)  
{unsigned char i;while(t--){for(i=0;i<12;i++);}
}

        7.1 DS18B20初始化

bit init_ds18b20(void)
{bit initflag = 0;DQ = 1;  Delay_OneWire(12);DQ = 0;  //拉低总线Delay_OneWire(80);DQ = 1;Delay_OneWire(10); //等待DS18B20拉低initflag = DQ;     //获取总线数据Delay_OneWire(5);return initflag;  
}

         7.2 向DS18B20写入一字节数据

void Write_DS18B20(unsigned char dat)
{unsigned char i;for(i=0;i<8;i++){DQ = 0;  //拉低DQ = dat&0x01; //获取dat最后一位数据Delay_OneWire(5);//写入数据DQ = 1;         //释放总线dat >>= 1;      //dat右移一位}Delay_OneWire(5);
}

        7.3 读取DS18B20数据寄存器的值

unsigned char Read_DS18B20(void)
{unsigned char i;unsigned char dat;//循环8次,读一个字节for(i=0;i<8;i++){DQ = 0;  //拉低dat >>= 1; DQ = 1;  //释放总线if(DQ)  // 因为dat>>1位默认是0 ,如果DQ为0,就不需要赋0了{dat |= 0x80; //给dat赋1}	    Delay_OneWire(5);}return dat;
}

        7.4 读取温度

unsigned int  get_temp()
{ unsigned int result;float i;unsigned char low,high;init_ds18b20();Write_DS18B20(0xcc);//跳过romWrite_DS18B20(0x44);//开始转换init_ds18b20();Write_DS18B20(0xcc);//跳过romWrite_DS18B20(0xbe);//开始转换low=Read_DS18B20(); //先读取低字节high=Read_DS18B20();//再读取高字节result =high&0x0f;  //获取高字节低4位数据(1位符号位,3位数据位)result<<=8;         //左移8位result=result|low;  //或上低字节,拼成16位数据。i=result*0.0625;    //为什么乘0.0625,下面讲result=i*100;       //温度值扩大100倍,方便数码管显示return result;      //返回温度值
}

这里为什么最后的结果result需要乘以*0.0625。我们上面讲过DS18B20的数据寄存器,其实高字节的低3位才是数据位。 如高字节的最后一位本来是 2^4,假如有这个数据。则数据寄存器这个位为1,因为我们已经左移了8位,则变成了2^8.则实际上扩大了2^8/2^4=2^4=16,比实际结果扩大了16倍,所以我们需要乘以0.0625,缩小16倍。这样才能得到真实值。

所以整体原因是因为result左移8位,导致数据扩大了16倍,所以result需要乘以0.0625缩小16倍!

8.结语 

 这次也是通过DS18B20的芯片手册来讲解如何读取DS18B20的数据,以及最后转化为真实温度。   最后也是用代码实例来巩固理论,以及验证理论。

        

http://www.lryc.cn/news/93473.html

相关文章:

  • set/map学习
  • JavaScript Web APIs学习总结
  • 萤石摄像头RTSP流获取(黑屏解决)
  • ThreadLocal引发的内存泄漏分析
  • 银行数据治理:数据质量管理实践
  • 2.7V至25V宽输入电压15A 峰值电流
  • Vue 父子组件应用指南:从基础到实战
  • todotodo
  • 创建autotool项目
  • 计算机概念
  • 【数学建模系列】TOPSIS法的算法步骤及实战应用——MATLAB实现
  • 网络安全(黑客)工具
  • 探究前后端数据交互方式
  • Yolov5轻量化:CVPR2023|RIFormer:无需TokenMixer也能达成SOTA性能的极简ViT架构
  • Spring-Retry实现及原理
  • Java中的锁
  • 学习系列:5种常见的单例模式变体及其实现方式
  • 三菱FX5U系列PLC之间进行简易PLC间链接功能的具体方法
  • 基于DBACAN的道路轨迹点聚类
  • 【项目】接入飞书平台
  • c++11 标准模板(STL)(std::ios_base)(三)
  • 在线协同办公小程序开发搭建开发环境
  • 【编译、链接、装载六】汇编——目标文件
  • 王道计算机考研408计算机组成原理汇总(下)
  • 偏向锁、轻量级锁、重量级锁、自旋锁、自适应自旋锁
  • Delta 一个新的 git diff 对比显示工具
  • C# 二进制序列化和反序列化示例
  • 【CSS】文字扫光 | 渐变光
  • Overhaul Distillation(ICCV 2019)原理与代码解析
  • <Linux开发>驱动开发 -之-内核定时器与中断