【STM32(HAL库)--DHT11温湿度传感器】

本文详细介绍了如何在STM32平台上使用DHT11温湿度传感器,通过CubeMX配置时钟、GPIO、串口和TIM定时器,以及利用HAL库编写代码实现数据读取、延时和错误处理,展示了从硬件到软件的完整过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

stm32控制DHT11温湿度传感器,进行温湿度的读取.以下是自己的一点学习过程和心得,若有不妥之处,还望各位大佬指正,在下感激不尽.

一.DHT11产品概述

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

二.数据访问

串行接口 (单线双向)

微处理器与 DHT11之间的通讯和同步,采用单总线数据格式,一次通讯时间4ms左右,数据分小数部分和整数部分,具体格式在下面说明,当前小数部分用于以后扩展,现读出为零.操作流程如下:
一次完整的数据传输为40bit,高位先出。
数据格式:
8bit湿度整数数据+ 8bit湿度小数数据+ 8bi温度整数数据+ 8bit温度小数数据+ 8bit校验和

数据传送正确时校验和数据等于“8bit湿度整数数据+8bit湿度小数数据+8bi温度整数数据+8bit温度小数数据” 所得结果的末8位
可见其数据读取是很简单的,下面来看看其时序图:
1.通讯过程如图1所示在这里插入图片描述

图1

总线空闲状态为高电平,主机把总线拉低等待DHT11响应,主机把总线拉低必须大于18毫秒(代码中一般为20毫秒),主机把总线拉高20-40毫秒,保证DHT11能检测到起始信号。 DHT11接收到主机的开始信号后,等待主机开始信号结束,由于是单总线,在主机发送起始信号要将DHT11的data引脚配至为输出模式,主机开始信号结束后,就要开始读数据,将DHT11的data引脚配至为输入模式.然后DHT111发送80us低电平响应信号.又发送80us高电平响应信号,即读取DHT11的响应信号,(主机发送开始信号后,可以切换到输入模式,或者输出高电平均可, 总线由上拉电阻拉高)在这里插入图片描述

总线为低电平,说明DHT11发送响应信号,DHT11发送响应信号后,再把总线拉高80us,准备发送数据,每一bit数据都以50us低电平时隙开始,高电平的长短定了数据位是0还是1.也就是通过读取电平是高,还是低且持续时间的长短来判断.

格式见下面图示.如果读取响应信号为高电平,则DHT11没有响应,请检查线路是否连接正常.当最后一bit数据传送完毕后, DHT11拉低总线50us,随后总线由上拉电阻拉高进入空闲状态。

数字0信号表示方法如图4所示
在这里插入图片描述

数字1信号表示方法.如图5所示
在这里插入图片描述

三.CubeMX配置

(1)时钟配置

如下图分别为设置HSE(高速外部时钟)以及时钟树的配置。选定HSE之后芯片会自动选定两个引脚用来连接外部晶振,设置LSE之后配置时钟树,设置HCLK为72MHz(最高72MHz,也可以配置其他),其配置图如图所示。
在这里插入图片描述
在这里插入图片描述

(2)调试接口配置

如图所示,将调试接口设置的设置为SW模式,占用芯片两个引脚。

在这里插入图片描述

(3)GPIO配置

如图,在CubeMX中芯片的引脚中点击鼠标左键可以给引脚设置功能。这里将pPF0设置为输出模式(由于DHT11只有一根数据线,所以其交互方式是半双工,也就是在运行时候需要动态改变该引脚的输入与输出方向,在这里只做简单的配置,其具体配置需要在代码里修改)。
在这里插入图片描述

(4)串口配置

为显示结果,用串口将转换结果传到电脑上,设置为异步模式,波特率为115200Bits/s,其UART配置如下图所示。设置完成之后会自动引出两个引脚用于串口通信。这里的串口,我在代码中进行了重定义
这里的串口,我在代码中进行了重定义,参考代码.

(5)TIM定时器配置

由于HAL库中没有微秒级延时函数,所以这里采用定时器取计数,达到微秒级延迟。设置定时器为内部时钟,由于设置的MCU主频为72MHz,所以这边设置(72-1)分频,这样就刚好是1MHz,也就达到了1us的时间,后续就由软件实现延时函数。
在这里插入图片描述

(6)引脚使用情况

本次除了调试接口和外部震荡接口外,还有PF0连接DHT11传感器的数据传输引脚.
在这里插入图片描述

四.CubeMX–HAL库代码

(1)TIM定时器实现us级延时,实现us级延迟

void delay_us(uint16_t nus)
{
	__HAL_TIM_SET_COUNTER(&htim1, 0);
	__HAL_TIM_ENABLE(&htim1);
	while (__HAL_TIM_GET_COUNTER(&htim1) < nus)
	{
	}
	__HAL_TIM_DISABLE(&htim1);
}

(2) 数据引脚方向

void DHT11_OUT(void )//输出引脚
{
	GPIO_InitStruct.Pin = GPIO_PIN_0;
  GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  HAL_GPIO_Init(GPIOF, &GPIO_InitStruct);
}
void DHT11_IN(void )//输入引脚
{
	GPIO_InitStruct.Pin = GPIO_PIN_0;
  GPIO_InitStruct.Mode = GPIO_MODE_INPUT ;
  GPIO_InitStruct.Pull = GPIO_PULLUP;
  GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
  HAL_GPIO_Init(GPIOF, &GPIO_InitStruct);
}

(3) DHT11的驱动函数(根据时序图来实现)

dht11.c文件


void DHT11_REST(void ) //主机发出起始信号
{
	DHT11_OUT();
	HAL_GPIO_WritePin (GPIOF ,GPIO_PIN_0 ,GPIO_PIN_RESET );
	HAL_Delay (20);
	HAL_GPIO_WritePin (GPIOF ,GPIO_PIN_0 ,GPIO_PIN_SET );
	delay_us (30);
}

//gpio端口为输入模式时,配置为上拉输入或者浮空输入,因外接上拉电阻,默认为高电平
//当有负信号输入时,gpio端口为1.
//若有负信号输入,当信号引脚连接到GND或者其他低电平信号时,GPIO端口会检测到低电平并显示为0。
//这是因为负信号的优先级高于上拉电阻的电平设置。


uint8_t DHT11_Check(void)
{
    uint8_t retry = 0;

    DHT11_IN();  // 设置数据引脚为输入模式

    while (HAL_GPIO_ReadPin(GPIOF, GPIO_PIN_0) == 1 && retry < 80)
    {
        retry++;
        delay_us(1);  // 延迟1微秒
    }

    if (retry >= 80)
        return 1;  // 如果在规定时间内引脚仍为高电平,表示传感器未响应,返回错误代码 1
    else
        retry = 0;

    while (HAL_GPIO_ReadPin(GPIOF, GPIO_PIN_0) == 0 && retry < 80)
    {
        retry++;
        delay_us(1);  // 延迟1微秒
    }

    if (retry >= 80)
        return 1;  // 如果在规定时间内引脚仍为低电平,表示传感器未响应,返回错误代码 1

    return 0;  // 传感器响应正常,返回成功代码 0
}
//读取一个位,参考高低电平的时序
uint8_t DHT11_Read_Bit(void)
{
	uint8_t retry=0;
	while((HAL_GPIO_ReadPin (GPIOF ,GPIO_PIN_0)==1) && (retry <100))
	{
		retry ++;
		delay_us (1);
	}
	retry=0;
	while((HAL_GPIO_ReadPin(GPIOF ,GPIO_PIN_0)==0) && (retry <100))
	{
		retry ++;
		delay_us (1);
	}
	delay_us (40);
	if(HAL_GPIO_ReadPin (GPIOF ,GPIO_PIN_0 )==1)
		return 1;// 返回读取到的位为高电平
	else 
		return 0;// 返回读取到的位为低电平
}

//读取一个字节,接收数据
uint8_t DHT11_Read_Byte(void)
{
	uint8_t dat=0;
	for(uint8_t i=0;i<8;i++)
	{
		dat <<= 1;
		dat |= DHT11_Read_Bit();
	}
	return dat;

}
//更据上面的数据格式,解码数据
uint8_t DHT11_Read_Data(uint8_t* humi, uint8_t* temp)
{
    uint8_t buf[5];  // 存储读取到的5个字节数据的缓冲区

    DHT11_REST();  // 初始化传感器通信

    if (DHT11_Check() == 0)  // 检查传感器是否正常响应
    {
        for (uint8_t i = 0; i < 5; i++)
            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;  // 读取数据成功,返回正常代码
}

(4) 函数的声明

dht11.h文件

#ifndef __dht11_h
#define __dht11_h

#include "main.h"

void DHT11_REST(void);                            //复位DHT11
uint8_t DHT11_Check(void);                       //DHT11状态反馈
uint8_t DHT11_Read_Bit(void);                    //读DHT11一位数据
uint8_t DHT11_Read_Byte(void);                   //读DHT11一字节数据
uint8_t DHT11_Read_Data(uint8_t* humi,uint8_t* temp);     //DHT11数据显示

#endif /*__dht11_h*/

(5) main中声明

	uint8_t temperature = 1;      //给一个初始化,用与判断               
	uint8_t humidity = 1; 
//上电先检测,电路是非有问题
DHT11_REST();                 //复位DHT11
  while(DHT11_Check())                           //检测DHT11连接
  {
	  printf ("No Connect!\r\n");
	  HAL_Delay(500);
  }
	printf ("Success!\r\n");

(6) 主函数

在while中

DHT11_Read_Data(&humidity,&temperature);
	
printf("temperature : %d ^C\r\n ",temperature) ;
printf ("humidity : %d %% \r\n",humidity);
HAL_Delay(100);

五.printf()函数重定义

在串口函数中加入下列代码,并在main()中加入#include “stdio.h”.

typedef struct __FILE FILE;

int fputc(int ch,FILE*stream)
{
	HAL_UART_Transmit(&huart1,(uint8_t *)&ch,1,0xFFFF);
	return ch;	
}

且要按下图操作,即可重定义完成
在这里插入图片描述

(六)声明

由于个人能力有限,若有不对之处,还望各位指正,在下感激不尽.

### 使用 DHT11 温湿度传感器配合 HAL 进行开发 #### 初始化和配置硬件资源 为了使 STM32 能够通过 GPIO 接口与 DHT11 通信,需先完成必要的初始化工作。这包括设置时钟、GPIO 和中断等外设参数。 ```c // 配置 GPIO 引脚用于连接 DHT11 的数据线 void MX_GPIO_Init(void) { __HAL_RCC_GPIOB_CLK_ENABLE(); // 启用 GPIOB 时钟 GPIO_InitTypeDef GPIO_InitStruct = {0}; /* 配置 PB5 作为 DHT11 数据输入/输出 */ GPIO_InitStruct.Pin = GPIO_PIN_5; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; // 设置推挽输出模式 GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); } ``` #### 编写函数来读取 DHT11 数据 接下来定义一个专门用来获取来自 DHT11 原始测量值的功能模块。此部分涉及到了具体协议细节以及位操作技巧[^3]。 ```c uint8_t dht_data[5]; int read_DHT11_Data() { uint8_t i,j; int retry=0; while(HAL_GPIO_ReadPin(DHT_PORT,DHT_PIN)==0 && retry<100)retry++; //等待DHT准备好发送数据 if(retry>=100)return -1; for(i=0;i<5;i++)dht_data[i]=0; //启动信号 HAL_GPIO_WritePin(DHT_PORT,DHT_PIN,GPIO_PIN_RESET); HAL_Delay(18); HAL_GPIO_WritePin(DHT_PORT,DHT_PIN,GPIO_PIN_SET); for (i=0;i<40;i++) { while(!HAL_GPIO_ReadPin(DHT_PORT,DHT_PIN)); //等待高电平到来 Delay_us(30); //延时一段时间后判断高低电平状态决定是0还是1 if(HAL_GPIO_ReadPin(DHT_PORT,DHT_PIN)) dht_data[i/8]|=(1<<(7-(i%8))); while(HAL_GPIO_ReadPin(DHT_PORT,DHT_PIN)); } return check_crc(); } bool check_crc(){ unsigned char crc=0; for(int i=0;i<4;i++){ crc+=dht_data[i]; } return crc==dht_data[4]?true:false; } ``` 上述代码实现了对 DHT11 设备的一次完整查询流程,并且包含了简单的校验机制以确保接收到的信息准确性。当一切正常运作之后,则可以进一步解析这些原始字节得到实际环境条件下的相对湿度百分比及摄氏度数值[^1]。 #### 解析并展示温湿度信息 最后一步是从之前获得的五个字节数组 `dht_data[]` 中提取有用的部分转换成人类可理解的形式打印出来或上传至云端平台。 ```c float humidity, temperature; if(read_DHT11_Data()==OK){ humidity=dht_data[0]+((float)dht_data[1])/10; temperature=dht_data[2]+((float)dht_data[3])/10; printf("Humidity=%.1f%%\n",humidity); printf("Temperature=%.1f°C\n",temperature); }else{ printf("Error reading data from sensor.\n"); } ``` 这段程序片段负责将二进制形式表示的湿度温度量转化为浮点型变量以便后续处理或是直接呈现给用户查看。
评论 15
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值