4.关于对DHT11单总线时序图的理解

关于对DHT11单总线时序图的理解

最近上网买了一个单总线的DHT11温湿度传感器,看了一些代码,这是自己的理解,并记录。

DHT11数字温湿度传感器资料

DHT11数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器。它应用专用的数字模块采集技术和温湿度传感技术,确保产品具有极高的可靠性与卓越的长期稳定性。传感器包括一个电阻式感湿元件和一个NTC测温元件,并与一个高性能8位单片机相连接。因此该产品具有品质卓越、超快响应、抗干扰能力强、性价比极高等优点。每个DHT11传感器都在极为精确的湿度校验室中进行校准。校准系数以程序的形式储存在OTP内存中,传感器内部在检测信号的处理过程中要调用这些校准系数。单线制串行接口,使系统集成变得简易快捷。超小的体积、极低的功耗,信号传输距离可达20米以上,使其成为各类应用甚至最为苛刻的应用场合的最佳选则。产品为4针单排引脚封装。连接方便,特殊封装形式可根据用户需求而提供。

串行接口(单线双向)DATA用于微处理器与DHT11之间的通讯和同步,采用单总线数据格式,一次通讯时间4ms左右,数据分小数部分和整数部分,具体格式在下面说明,当前小数部分用于以后扩展,现读出为零.操作流程如下:

一次完整的数据传输为40bit,高位先出。

数据格式

数据格式:8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据+8bit

校验数据

校验和数据传送正确时校验和数据等于“8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据”所得结果的末8位。

用户MCU发送一次开始信号后,DHT11从低功耗模式转换到高速模式,等待主机开始信号结束后,DHT11发送响应信号,送出40bit的数据,并触发一次信号采集,用户可选择读取部分数据.从模式下,DHT11接收到开始信号触发一次温湿度采集,如果没有接收到主机发送开始信号,DHT11不会主动进行温湿度采集.采集数据后转换到低速模式。

理解

原子有现成的代码可以直接用,但是直接用没意思啊,读数据看不懂也不行啊,然后自己花了点时间琢磨了一下,关于读写这一个温湿度传感器的过程
io口的配置代码我就不放上来了,只放一些关键的
io方向设置,由于是单总线,数据的输入输出都用一个io口就可以实现

//IO方向设置
#define DHT11_IO_IN()  {GPIOG->MODER&=~(3<<(9*2));GPIOG->MODER|=0<<9*2;}	//PG9输入模式
#define DHT11_IO_OUT() {GPIOG->MODER&=~(3<<(9*2));GPIOG->MODER|=1<<9*2;} 	//PG9输出模式 
IO操作函数											   
#define	DHT11_DQ_OUT PGout(9) //数据端口	PG9 
#define	DHT11_DQ_IN  PGin(9)  //数据端口	PG9 

in 输入模式
out 输出模式

先进行通讯,也就是检查有没有DHT11模块

首先主机发送开始信号,即:拉低数据线,保持 t1(至少 18ms)时间,然后拉高数据线 t2
(20~40us)时间,然后读取 DHT11 的响应,正常的话,DHT11 会拉低数据线,保持 t3 (40~50us)
时间,作为响应信号,然后 DHT11 拉高数据线,保持 t4(40~50us)时间后,开始输出数据。。
在这里插入图片描述

//复位DHT11
void DHT11_Rst(void)	   
{                 
	DHT11_IO_OUT(); 	//SET OUTPUT
  DHT11_DQ_OUT=0; 	//拉低DQ
  delay_ms(20);    	//拉低至少18ms
  DHT11_DQ_OUT=1; 	//DQ=1 
	delay_us(30);     	//主机拉高20~40us
}

//等待DHT11的回应
//返回1:未检测到DHT11的存在
//返回0:存在
u8 DHT11_Check(void) 	   
{   
	u8 retry=0;
	DHT11_IO_IN();//SET INPUT	 
    while (DHT11_DQ_IN&&retry<100)//DHT11会拉低40~80us
	{
		retry++;
		delay_us(1);
	};	 
	if(retry>=100)return 1;
	else retry=0;
    while (!DHT11_DQ_IN&&retry<100)//DHT11拉低后会再次拉高40~80us
	{
		retry++;
		delay_us(1);
	};
	if(retry>=100)return 1;	    
	return 0;
}

总线为低电平,说明DHT11发送响应信号,DHT11发送响应信号后,再把总线拉高80us,准备发送数据,每一bit数据都以50us低电平时隙开始,高电平的长短定了数据位是0还是1.格式见下面图示.如果读取响应信号为高电平,则DHT11没有响应,请检查线路是否连接正常.当最后一bit数据传送完毕后,DHT11拉低总线50us,随后总线由上拉电阻拉高进入空闲状态。

DHT11 输出数字‘0’的时候

DHT11 先拉低50us程序应该在第二个while
处运行,当跳出这个循坏后,此时应该进入高电平阶段 时间持续26-28us 但是
延时程序延时了40us 此时DHT11_DQ_IN为0 返回“0”,
时间为50+40us 但此时时序已经在下一个bit开始处的10个us左右,为低电平
在这里插入图片描述

//从DHT11读取一个位
//返回值:1/0
u8 DHT11_Read_Bit(void) 			 
{
 	u8 retry=0;
	while(DHT11_DQ_IN&&retry<100)//等待变为低电平
	{
		retry++;
		delay_us(1);
	}
	retry=0;
	while(!DHT11_DQ_IN&&retry<100)//等待变高电平
	{
		retry++;
		delay_us(1);
	}
	delay_us(40);//等待40us
	if(DHT11_DQ_IN)return 1;
	else return 0;		   
}

DHT11 输出数字‘1’的时候

由上面知道,这一个bit 已经运行到了10us左右的位置
此时运行到了第二个While里面运行直到结束后,
来了个70us 高电平持续 此时正在运行delay 40us 用完之后
距离高电平结束还有30个us左右,也即是说这时候检测DHT11_DQ_IN 视为“1” 的 此时返回“1”
在这里插入图片描述
到这里其实就已经完成了两次, 分别对0和1数据的读取,读取8次为一个字节

//从DHT11读取一个字节
//返回值:读到的数据
u8 DHT11_Read_Byte(void)    
{        
    u8 i,dat;
    dat=0;
	for (i=0;i<8;i++) 
	{
   		dat<<=1; 
	    dat|=DHT11_Read_Bit();
    }						    
    return dat;
}

然后总共读取五个字节 最后一个字节是对前面的校验

//从DHT11读取一次数据
//temp:温度值(范围:0~50°)
//humi:湿度值(范围:20%~90%)
//返回值:0,正常;1,读取失败
u8 DHT11_Read_Data(u8 *temp,u8 *humi)    
{        
 	u8 buf[5];
	u8 i;
	DHT11_Rst();
	if(DHT11_Check()==0)
	{
		for(i=0;i<5;i++)//读取40位数据
		{
			buf[i]=DHT11_Read_Byte();
		}
		if((buf[0]+buf[1]+buf[2]+buf[3])==buf[4])
		{
			*humi=buf[0];
			*temp=buf[2];
		}
	}else return 1;
	return 0;	    
}
### DHT11 时序图与数据通信协议解析 #### 协议概述 DHT11 是一种温湿度传感器,采用单总线通信协议完成主机与设备之间的数据交互。该协议通过特定的时序控制实现数据的可靠传输[^1]。 #### 主机发起请求过程 主机向 DHT11 发起读取操作前,需先将总线拉低至少 **18ms** 来触发传感器的响应机制。之后主机释放总线并转入输入模式,等待 DHT11 的应答信号。DHT11 收到启动信号后会回应一个约 **80μs** 的低电平信号表示已准备好发送数据[^2]。 #### 响应信号处理 在接收到 DHT11 的应答信号后,主机需要检测总线的状态变化。具体而言,在 DHT11 将总线重新拉高大约 **80μs** 后,它会再次将其拉低以开始逐 bit 地发送数据。此时每比特数据均以一段固定的 **50μs** 低电平时隙作为开头,而后续高电平持续的时间决定了当前位是逻辑 `0` 还是逻辑 `1`: - 如果高电平时间为 **26~28μs**,则代表逻辑 `0`; - 若高电平时间延长至 **70μs** 左右,则对应逻辑 `1`[^3]。 #### 数据帧结构 每次完整的数据交换由 40 bits 组成,分为整数部分和小数部分两组分别描述温度与相对湿度的信息。其中前两个字节依次存储湿度的整数值及其小数补充;接着第三、四个字节记录的是摄氏度下的实际测量值以及相应的小数修正项;最后一位用作校验码验证整个报文的有效性[^2]。 以下是基于上述原理的一个典型代码片段展示如何捕获来自 DHT11 的原始二进制流: ```c void read_DHT11() { unsigned char i,j; unsigned int value=0; // 初始化 GPIO 及配置为主输出方向 DHT11_GPIO_Mode_OUT(); // 下拉总线超过 18 ms 触发采集周期 DATA_GPIO_OUT_LOW(); delay_ms(19); // 上升沿结束命令阶段切换回接收端口监听外部反馈动作 DATA_GPIO_OUT_HIGH(); delay_us(30); DHT11_GPIO_Mode_IN(); // 检测是否有有效回复标志到来 if(!DATA_GPIO_IN){ while(!DATA_GPIO_IN); // 等待上升沿确认握手成功 while(DATA_GPIO_IN); // 随即下降沿标志着正式进入资料区段 } for(i=0;i<5;i++) { // 循环五次获取全部五个八位元串列 for(j=0;j<8;j++) { while(!DATA_GPIO_IN); // 跳过初始固定长度短脉冲间隔 delay_us(40); if(DATA_GPIO_IN) // 判断剩余高位保持期间是否超出标准界限从而区分零壹定义 value |= (0x80>>(j)); while(DATA_GPIO_IN); // 完全消耗掉本次循环单元直至下一个低位起点为止 } value <<=8; // 移动寄存器位置以便累积下一组新加入成员 } } ``` #### 总结 通过对以上内容的学习可以发现,尽管目前设定下单位时间内所能承载的最大信息量有限,但由于其实现简单稳定所以在许多低成本应用场景仍然广泛适用。然而随着技术进步未来或许能够探索更多改进空间来进一步提升效率或者兼容更复杂的功能需求。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值