单片机学习笔记---DS18B20温度读取

目录

OneWire.c

模拟初始化的时序

模拟发送一位的时序

 模拟接收一位的时序

模拟发送一个字节的时序

模拟接收一个字节的时序

OneWire.h

DS18B20.c

DS18B20数据帧

模拟温度变换的数据帧

模拟温度读取的数据帧

DS18B20.h

main.c


上一篇讲了DS18B20温度传感器的工作原理,这节开始代码演示!

新创建一个工程:DS18B20温度读取

将前面我们学过的几个模块化代码添加进来

然后创建main.c,DS18B20.c,DS18B20.h,OneWire.c和OneWire.h文件

开始代码讲解:

OneWire.c

首先我们根据原理图定义引脚

#include <REGX52.H>

//引脚定义
sbit OneWire_DQ=P3^7;

我们再根据上一篇讲的时序逐个定义函数写在OneWire.c里面 

模拟初始化的时序

初始化:主机将总线拉低至少480us,然后释放总线,等待15~60us(可以取中间值)后,存在的从机会拉低总线60~240us(可以取中间值)以响应主机,之后从机将释放总线

unsigned char OneWire_Init(void)
{
	unsigned char i;
	unsigned char AckBit;//返回值,应答位
	OneWire_DQ=1;//拉低之前确保是释放状态
    //因为我们在执行任何操作的时候都可以用初始化来打断它,
    //所以总线再初始化的时候还是有可能处于0的状态,所以先把它拉高再拉低
	OneWire_DQ=0;

	i = 247;while (--i);		//Delay 500us
	OneWire_DQ=1;//释放

    //等待15~60us后,存在的从机会拉低总线60~240us以响应主机
	i = 32;while (--i);			//Delay 70us
    //我们直接延时70us后肯定已经到了从机拉低总线的状态了

    //然后直接可以读了
	AckBit=OneWire_DQ;//把IO口电平读出来赋值给应答位
    
    //初始化时有两个部分(复位和响应),
    //每一部分时序至少480us,这部分我们已经Delay了70微秒,
    //为了将这段时序走完,再Delay 500us,那么这个时序肯定就走完了
	i = 247;while (--i);		//Delay 500us
	
    //最后是从机将释放总线,我们主要写主机的代码,
    //不体现从机的代码,所以不用管最后释放的这一步

    return AckBit;//返回应答位
}

 怎么产生Delay 500us的那一行代码呢?

 

但因为我们初始化的条件说的是主机将总线拉低至少480us,为了保险起见,我们最好Delay500us

将500微秒的主体部分复制过来就是我们代码里的这部分:

模拟发送一位的时序

发送一位:主机将总线拉低60~120us(最大不能超过120us),然后释放总线,表示发送0;主机将总线拉低1~15us,然后释放总线,表示发送1。从机将在总线拉低30us后(典型值)读取电平,整个时间片应大于60us

void OneWire_SendBit(unsigned char Bit)
{
	unsigned char i;
    //在这一步之前先给它置1也行,但是考虑到初始化后它一定是1了
	OneWire_DQ=0;//所以直接拉低就行

    //主机将总线拉低60~120us,然后释放总线,表示发送0;
    //主机将总线拉低1~15us,然后释放总线,表示发送1
    //因此我们直接在10微秒的时候把bit放在线上
    //如果是0的话它肯定一直都是0,不会变化
    //如果是1的话它就会变成高位
	i = 4;while (--i);			//Delay 10us
    //由于调用一个函数就已经用了4us,所以我们直接生成一个14us的函数
	OneWire_DQ=Bit;//把bit放在线上
    
    //从机将在总线拉低30us后(典型值)读取电平,整个时间片应大于60us
    //我们已经Delay了10us,(4us是调用一个函数的时间)
    //再生成一个Delay 54us的代码就走完了时序(4us是调用一个函数的时间)
	i = 24;while (--i);			//Delay 50us
	
    OneWire_DQ=1;//释放
}

 模拟接收一位的时序

接收一位:主机将总线拉低1~15us,然后释放总线,并在拉低后15us内读取总线电平(尽量贴近15us的末尾),读取为低电平则为接收0,读取为高电平则为接收1 ,整个时间片应大于60us

unsigned char OneWire_ReceiveBit(void)
{
	unsigned char i;
	unsigned char Bit;
	OneWire_DQ=0;//拉低
    //Delay 5us就生成9us的代码(因为调用函数就是4us了)
    //在这段时间里从机接着也拉低总线(我们写是主机的代码,不体现从机)
	i = 2;while (--i);			//Delay 5us
	OneWire_DQ=1;//释放
    
    //因为释放需要一定的时间,所以需要Delay
    //如果释放后立马读的话可能来不及恢复变成1
	i = 2;while (--i);			//Delay 5us

	Bit=OneWire_DQ;//读取并赋给Bit
    
	i = 24;while (--i);			//Delay 50us
	return Bit;//返回Bit
}

模拟发送一个字节的时序

发送一个字节:连续调用8次发送一位的时序,依次发送一个字节的8位(低位在前)

void OneWire_SendByte(unsigned char Byte)
{
	unsigned char i;
	for(i=0;i<8;i++)
	{
		OneWire_SendBit(Byte&(0x01<<i));//从低位到高位
	}
}

模拟接收一个字节的时序

接收一个字节:连续调用8次接收一位的时序,依次接收一个字节的8位(低位在前)

unsigned char OneWire_ReceiveByte(void)
{
	unsigned char i;
	unsigned char Byte=0x00;
	for(i=0;i<8;i++)
	{
		if(OneWire_ReceiveBit()){Byte|=(0x01<<i);}//从低位到高位
	}
	return Byte;
}

OneWire.h

在OneWire.c声明一下这些函数

#ifndef __ONEWIRE_H__
#define __ONEWIRE_H__

unsigned char OneWire_Init(void);
void OneWire_SendBit(unsigned char Bit);
unsigned char OneWire_ReceiveBit(void);
void OneWire_SendByte(unsigned char Byte);
unsigned char OneWire_ReceiveByte(void);

#endif

DS18B20.c

我们要在这文件中分别调用前面我们写好的函数,完成上一篇博客讲的这个数据帧

先定义一下我们要用到的三个指令:跳过ROM,温度变换,读暂存器。

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

//DS18B20指令
#define DS18B20_SKIP_ROM			0xCC
#define DS18B20_CONVERT_T			0x44
#define DS18B20_READ_SCRATCHPAD 	0xBE

DS18B20数据帧

模拟温度变换的数据帧

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

void DS18B20_ConvertT(void)
{
	OneWire_Init();
	OneWire_SendByte(DS18B20_SKIP_ROM);
	OneWire_SendByte(DS18B20_CONVERT_T);
}

模拟温度读取的数据帧

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

除了写温度读取的数据帧之外,我们还要把读出来的两个温度字节合成16位并转换成十进制的温度值

float DS18B20_ReadT(void)
{
	unsigned char TLSB,TMSB;//暂存器的第一个字节和第二个字节
	int Temp;//中间变量
	float T;//T表示实际温度,float型既可以表示正负也可以表示小数
	OneWire_Init();
	OneWire_SendByte(DS18B20_SKIP_ROM);
	OneWire_SendByte(DS18B20_READ_SCRATCHPAD);
    //一旦发完这个指令,总线的控制权就交给从机了

	TLSB=OneWire_ReceiveByte();
	TMSB=OneWire_ReceiveByte();
    //这样就把暂存器里前两个字节(温度字节的数据)给读出来了

    //转换温度
	Temp=(TMSB<<8)|TLSB;//两个字节合成16位
    //TMSB<<8将第二个字节放到高位上

    //无符号的TLSB,TMSB合成16位数后本身包含了符号位
    //无符号的转换成有符号的int内容没有改变
    //我们可以通过上面的示例图看到合成16位后bit4才是温度整数值的最低位
    //二进制数向左挪一位和乘以2是一样的,往左挪4位就是乘以16
    //因为实际值的最后一位代表2的-4次方,变为2的0次方相差16倍
    //为了不损失精度,就将Temp往右挪4位,转换成实际温度就除以16


    //如果还是看不懂的话,可以用十进制来做个类比:比如本来是1.0001结果把小数点擦掉饿了,
    //是不是成了10001,这就扩大了一万倍,想要准确的话就要除以1000

    //这就可以理解了,我们将两个字节拼成了16位的二进制数据之后赋给int型的Temp,
    //那么我们16就都成了整数了,
    //为了保证最后四位还是我们的小数位,就要将小数点往左移4位,那么除以几呢?
    //2的-4次方到2的0次方相差16倍,相当于Temp比实际的温度值T扩大了16倍,想要得到T那就是除以16

	T=Temp/16.0;
	return T;
}

DS18B20.h

声明一下这两个函数

#ifndef __DS18B20_H__
#define __DS18B20_H__

void DS18B20_ConvertT(void);
float DS18B20_ReadT(void);

#endif

main.c

先定义一个实际温度值T

#include <REGX52.H>
#include "LCD1602.h"
#include "DS18B20.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)				//如果温度小于0
		{
			LCD_ShowChar(2,1,'-');	//显示负号
			T=-T;			//将温度变为正数
		}
		else				//如果温度大于等于0
		{
			LCD_ShowChar(2,1,'+');	//显示正号
		}
		LCD_ShowNum(2,2,T,3);		//显示温度整数部分,浮点型转换整型自动忽略小数位
		LCD_ShowChar(2,5,'.');		//显示小数点
		LCD_ShowNum(2,6,(unsigned long)(T*10000)%10000,4);//显示温度小数部分
        //T*10000将小数挪成整数部分
        //(unsigned long)(T*10000)强制类型转换成(unsigned long),因为它超过了unsigned int
        //小数部分丢弃并且能够取余了
        //(T*10000)%10000得到10000的后四位
	}
}

效果请看视频:

DS18B20温度读取

以上就是本节的内容,源码会放在评论区,有问题可以评论区留言!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/387130.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

imazing怎么连接苹果手机

imazing怎么连接苹果手机 要连接苹果手机&#xff0c;您可以选择使用数据线或无线网络&#xff08;Wi-Fi&#xff09;两种方式。以下是具体的步骤&#xff1a; 使用数据线连接&#xff1a; 准备工具&#xff1a;确保您的Mac或Windows电脑已经安装了iMazing软件&#xff0c;并且…

【十八】【C++】deque双端队列简单使用和deque底层实现探究(部分代码)

deque简单使用 在C中&#xff0c;双端队列&#xff08;Double-Ended Queue, deque&#xff09;是一种具有动态大小的序列容器&#xff0c;允许在两端快速插入和删除元素。与std::vector相比&#xff0c;std::deque提供了更加灵活的数据结构&#xff0c;特别是在需要频繁在序列…

基于Transformer的机器学习模型的主动学习

主动学习和基于Transformer的机器学习模型的结合为有效地训练深度学习模型提供了强有力的工具。通过利用主动学习&#xff0c;数据科学家能够减少训练模型所需的标记数据的数量&#xff0c;同时仍然达到高精度。本文将探讨基于Transformer的机器学习模型如何在主动学习环境中使…

Java图形化界面编程—— ImageIO 笔记

2.8.4 ImageIO的使用 在实际生活中&#xff0c;很多软件都支持打开本地磁盘已经存在的图片&#xff0c;然后进行编辑&#xff0c;编辑完毕后&#xff0c;再重新保存到本地磁盘。如果使用AWT要完成这样的功能&#xff0c;那么需要使用到ImageIO这个类&#xff0c;可以操作本地磁…

数字图像处理实验记录十(图像分割实验)

一、基础知识 1、什么是图像分割 图像分割就是指把图像分成各具特性的区域并提取出感兴趣目标的技术和过程&#xff0c;特性可以是灰度、颜色、纹理等&#xff0c;目标可以对应单个区域&#xff0c;也可以对应多个区域。 2、图像分割是怎么实现的 图像分割算法基于像素值的不连…

Leetcode 115 不同的子序列

题意理解&#xff1a; 给你两个字符串 s 和 t &#xff0c;统计并返回在 s 的 子序列 中 t 出现的个数&#xff0c;结果需要对 109 7 取模。 即此题可以理解为&#xff1a;从s中删除元素去构造t,有多少种方法 或者也可以理解为&#xff1a;s中按顺序取t,有多少个 则一定有s和t…

HGAME 2024 WEEK2 Web方向题解 全

---------【WEEK-2】--------- What the cow say? 题目描述&#xff1a;the cow want to tell you something 注意title&#xff0c;Python的flask漏洞可多呢 版本310 先测一下SSTI 正常情况下 SSTI测试 变量渲染测试&#xff0c;被waf了&#xff0c;说明方向对了 单单过滤…

算法学习——LeetCode力扣回溯篇1

算法学习——LeetCode力扣回溯篇1 77. 组合 77. 组合 - 力扣&#xff08;LeetCode&#xff09; 描述 任何顺序 返回答案。 示例 示例 1&#xff1a; 输入&#xff1a;n 4, k 2 输出&#xff1a; [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], ] 示例 2&#xff1a; 输…

Leetcode 606.根据二叉树创建字符串

给你二叉树的根节点root&#xff0c;请你采用前序遍历的方式&#xff0c;将二叉树转化为一个由括号和整数组成的字符串&#xff0c;返回构造出的字符串。 空节点使用一对空括号对"root"表示&#xff0c;转化后需要省略所有不影响字符串与原始二叉树之间的一对一映射…

openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络

文章目录 openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络219.1 查看网络状况 openGauss学习笔记-219 openGauss性能调优-确定性能调优范围-硬件瓶颈点分析-网络 获取openGauss节点的CPU、内存、I/O和网络资源使用情况&#xff0c;确认这些资源…

在Meteor Lake平台上使用NPU进行AI推理加速

在Meteor Lake平台上&#xff0c;英特尔通过神经处理单元 (NPU) 将人工智能直接融入芯片中&#xff0c;实现桌面电脑平台的AI推理功能。神经处理单元 (NPU) 是一种专用人工智能引擎&#xff0c;专为运行持续的人工智能推理工作负载而设计。与即将推出的支持深度人工智能集成的 …

【MySQL】索引事务

MySQL索引事务 1. 索引1.1 概念1.2 作用1.3 使用场景1.4 使用1.5 案例 2. 事务2.2 事物的概念2.3 使用 3. 内容重点总结 1. 索引 1.1 概念 索引是一种特殊的文件&#xff0c;包含着对数据表里所有记录的引用指针。可以对表中的一列或多列创建索引&#xff0c; 并指定索引的类…

ClickHouse--04--数据库引擎、Log 系列表引擎、 Special 系列表引擎

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 1.数据库引擎1.1 Ordinary 默认数据库引擎1.2 MySQL 数据库引擎MySQL 引擎语法字段类型的映射 2.ClickHouse 表引擎3.Log 系列表引擎几种 Log 表引擎的共性是&#…

Golang快速入门到实践学习笔记

Go学习笔记 1.基础 Go程序设计的一些规则 Go之所以会那么简洁&#xff0c;是因为它有一些默认的行为&#xff1a; 大写字母开头的变量是可导出的&#xff0c;也就是其它包可以读取 的&#xff0c;是公用变量&#xff1b;小写字母开头的就是不可导出的&#xff0c;是私有变量…

Python算法题集_二叉树的最大深度

Python算法题集_二叉树的最大深度 题104&#xff1a;二叉树的最大深度1. 示例说明2. 题目解析- 题意分解- 优化思路- 测量工具 3. 代码展开1) 标准求解【DFS自顶向下】2) 改进版一【DFS自底向上】3) 改进版二【BFS】 4. 最优算法 本文为Python算法题集之一的代码示例 题104&am…

现代化端口扫描工具RustScan

今天是大年初五&#xff0c;喜迎财神 &#xff0c;祝大家✔️顺风顺水 ✔️诸事如意 ✔️财源滚滚 ✔️大吉大利 顺便提一下&#xff0c;老苏的博客启用了新域名&#xff1a; https://laosu.tech 什么是 RustScan &#xff1f; RustScan 是一款现代化的端口扫描器。能快速找到端…

数学实验第三版(主编:李继成 赵小艳)课后练习答案(九)(1)(2)

实验九&#xff1a;线性函数极值求解 练习一 1.求解线性规划问题&#xff1a; &#xff08;1&#xff09;max z3,s.t. clc;clear; %采用软件解法 c[-3,-1]; a[-1,1;1,-2;3,2]; b[2;2;14]; [x,min]linprog(c,a,b)找到最优解。 x 4 1 min -13 题上要求的是最大值&#…

【从零到Offer】MySQL最左匹配

前言 ​ 相信大家在日常开发时&#xff0c;也经常能听到“最左匹配”这个词&#xff0c;那么什么是最左匹配呢&#xff1f;本篇文章就带你一起探索“最左匹配”的神奇秘密。 什么是最左匹配 ​ 最左匹配&#xff0c;通常指的是最左前缀匹配原则&#xff0c;即MySQL在检索数据…

c++ Qt 数据库操作

1、准备工作 Qt本身并没有数据库功能&#xff0c;但是Qt支持调用其他主流的数据库产品&#xff0c;并且这些数据库产品统一了Qt的接口&#xff0c;实际上是一种数据库的中间件。 Qt支持以下数据库类型&#xff1a; 嵌入式常用的数据库是sqlite3&#xff0c;本体只有几兆大小。非…

UnityShader玉石效果

效果&#xff1a; 代码&#xff1a; Shader "MyShader/Jade" {Properties{_DiffuseColor("漫反射颜色",color)(1,1,1,1)_ThicknessMap("厚度图",2d)"white"{}_AddColor("叠加颜色",color)(1,1,1,1)_CubeMap("环境贴图…