STM32-HAL库驱动DHT11温湿度传感器 --2024.9.28

目录

一、教程简介

二、驱动原理讲解

(一)通信4步骤

(二)传感器数据解析

三、CubeMX生成底层代码

(一)基础配置

(二)配置DHT11的驱动引脚

(三)配置串口

四、Keil中编写代码

[(一)dht11.c 代码](#(一)dht11.c 代码)

[(二)dht11.h 代码](#(二)dht11.h 代码)

[(三)main.c 中调用](#(三)main.c 中调用)

五、效果展示


一、教程简介

DHT11是单片机开发常用的一个温湿度传感器,采用单总线通信,优点是单片机和传感器的连接只需要一根数据线,缺点则是对通信时序的要求较高。

本教程用通俗易懂的语言和详细的操作过程截图,为开发者清除DHT11这只拦路虎,本教程还提供可以快速使用DHT11的驱动代码,只要跟着本教程操作,都可以正确读取到温湿度信息。

二、驱动原理讲解

DHT11采用的是单总线的通信方式,系统中数据的交换、控制均由单总线完成。(注意:DHT11的数据引脚需要一个4.7K的上拉电阻,若使用的传感器是不带PCB的那种,请自己外加上拉电阻)。

(一)通信4步骤

步骤一:

DHT11上电后(DHT11上电后要等待1S以越过不稳定状态在此期间不能发送任何指令),测试环境温湿度数据,并记录数据,同时DHT11的DATA数据线由上拉电阻拉高一直保持高电平;此时DHT11的DATA引脚处于输入状态,时刻检测外部信号。

步骤二:

微处理器的I/0设置为输出同时输出低电平,且低电平保持时间不能小于18ms(最大不得超过30ms),然后微处理器的I/0设置为输入状态,由于上拉电阻,微处理器的I/0即DHT11的DATA数据线也随之变高,等待DHT11作出回答信号。发送信号如下图所示:

步骤三:

DHT11的DATA引脚检测到外部信号有低电平时,等待外部信号低电平结束,延迟后DHT11的DATA引脚处于输出状态,输出83微秒的低电平作为应答信号,紧接着输出87微秒的高电平通知外设准备接收数据,微处理器的I/0此时处于输入状态,检测到I/0有低电平(DHT11回应信号)后,等待87微秒的高电平后的数据接收,发送信号如图5所示:

步骤四:

由DHT11的DATA引脚输出40位数据,微处理器根据I/0电平的变化接收40位数据,位数据"0"的格式为:54微秒的低电平和23-27微秒的高电平,位数据"1"的格式为:54微秒的低电平加68-74微秒的高电平。位数据"0"、"1"格式信号如图6所示:

结束信号

DHT11的DATA引脚输出40位数据后,继续输出低电平54微秒后转为输入状态,由于上拉电阻随之变为高电平。但DHT11内部重测环境温湿度数据,并记录数据,等待外部信号的到来。

(二)传感器数据解析

传感器发送的40位数据分为5个部分,分别是:湿度高8位、湿度低8位、温度高8位、温度低8位、校验位。下面举例分析:

示例一:接收到的40位数据为

|-----------|-----------|-----------|-----------|-----------|
| 0011 0101 | 0000 0000 | 0001 1000 | 0000 0100 | 0101 0001 |
| 湿度高8位 | 湿度低8位 | 温度高8位 | 温度低8位 | 校验位 |

计算:

00110101 + 00000000 + 00011000 + 00000100 = 01010001

校验正确,接收数据正确。

湿度:00110101(整数) = 0x35 = 53% ,湿度小数为0。

所以湿度为: 53%

温度:00011000(整数) = 0x18 = 24 度 ,00000100(小数) = 0x04 = 0.4度

所以温度为:24 + 0.4 = 24.4 摄氏度

三、CubeMX生成底层代码

(一)基础配置

1、配置Debug

2、配置外部高速晶振

3、 配置时钟

(二)配置DHT11的驱动引脚

将任意一个引脚配置为:输出模式、内部上拉、高速模式、重命名为DHT11

(三)配置串口

(四)生成工程文件

四、Keil中编写代码

(一)dht11.c 代码

cpp 复制代码
#include "dht11.h"
/*
 * DHT11引脚:输入/输出模式配置函数 
 * Mode = 0/INPUT  时 输入模式  
 * Mode = 1/OUTPUT 时 输出模式  
 */
void DHT11_PIN_Mode(int Mode)
{	
   if(Mode)   
   {
		GPIO_InitTypeDef GPIO_InitStruct = {0};							// 定义GPIO_InitTypeDef结构体 
		GPIO_InitStruct.Pin = DHT11_Pin;                    // 引脚选择
		GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;         // 引脚模式:输出模式
		GPIO_InitStruct.Pull = GPIO_NOPULL;                 // 配置内部上拉
		GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_HIGH;       // 引脚速率:高速
		HAL_GPIO_Init(DHT11_GPIO_Port, &GPIO_InitStruct);
   }
   else
   {
	    GPIO_InitTypeDef GPIO_InitStruct = {0};						// 定义GPIO_InitTypeDef结构体 
		GPIO_InitStruct.Pin = DHT11_Pin;				    				// 引脚选择
		GPIO_InitStruct.Mode = GPIO_MODE_INPUT;							// 引脚模式:输入模式
		GPIO_InitStruct.Pull = GPIO_NOPULL;									// 配置内部上拉
		HAL_GPIO_Init(DHT11_GPIO_Port, &GPIO_InitStruct);
   }
}  

/*
 *DHT11起始函数
 *根据DHT11时序图,主机要要发送起始信号,需要将总线电平拉低(18~30ms)
 */
void DHT11_Start(void)
{
	DHT11_PIN_Mode(OUTPUT);
	DHT11_IO_SET;    				// 先让总线处于高电平状态
	HAL_Delay(1);
	DHT11_IO_RESET;  				// 拉低总线20ms,表示主机发送起始信号
	HAL_Delay(20);
	DHT11_IO_SET;    				// 将总线拉高等待传感器响应
	DHT11_Delay_us(30);
}

/**
  * DHT11响应检测函数
  * 返回1:未检测到DHT11的存在
  * 返回0:存在出现由高到低的变化即可
  */
uint8_t DHT11_Check(void)
{
	uint8_t retry = 0;
	DHT11_PIN_Mode(INPUT);              //将引脚切换为输入模式
	while(!DHT11_IO_Read && retry<100)  //单片机发送起始信号后,DHT11会将总线拉低83微妙
	{
		retry++;
		DHT11_Delay_us(1);
	}
	if(retry >= 100)return 1;
	else retry = 0;
	
	while(DHT11_IO_Read && retry<100)  //DHT11拉低后会再次拉高87微妙
	{
		retry++;
		DHT11_Delay_us(1);
	}
	if(retry >= 100) return 1;
	return 0;
}


/**
  * 从DHT11读取一个位
  * 返回值:1/0
  */
uint8_t DHT11_Read_Bit(void)
{
	DHT11_PIN_Mode(INPUT);
	while(!DHT11_IO_Read);
	DHT11_Delay_us(40);
	if(DHT11_IO_Read)
	{
		while(DHT11_IO_Read);
		return 1;
	}
	else
	{
		return 0;
	}
}

/**
  *  读取一个字节数据 1byte / 8bit
  *  返回值是一个字节的数据
  */
uint8_t DHT11_Read_Byte(void)
{
    uint8_t i,buf = 0;                             //  暂时存储数据
    
    for(i=0; i<8 ;i++)
    {
        buf <<= 1;                                 
        if(DHT11_Read_Bit())                        //  1byte -> 8bit
        {
            buf |= 1;                              //  0000 0001
        }
    }
    return buf;
}

/**
  * 读取温湿度传感器数据 5byte / 40bit
  * 使用方法:创建两个float变量,将变量地址传入函数
  * 注意:两次使用该函数的间隔需要大于2秒,否则会导致数据测量不准确
  */
uint8_t data[5] = {0};

uint8_t DHT11_READ_DATA(float *temp, float *humi)
{
   uint8_t i;
   DHT11_Start();                                 //  主机发送启动信号
   
   if(!DHT11_Check())                             //  如果DHT11应答     
   {  
      for(i=0; i<5; i++)
      {                        
         data[i] = DHT11_Read_Byte();             //  读取 5byte
      }
      if(data[0] + data[1] + data[2] + data[3] == data[4])
      {
				 *humi = data[0] + 0.1*data[1];
				 *temp = data[2] + 0.1*data[3];
         return 1;                                //  数据校验通过
      }
      else return 0;                              //  数据校验失败
   }
   else return 2;                                 //  如果DHT11不应答
}

/**
  * 微妙延时函数
  * 全系列通用,只需要将宏定义CPU_FREQUENCY_MHZ根据时钟主频修改即可。
  * 系统滴答定时器是HAL库初始化的,且必须有HAL库初始化。
  */
#define CPU_FREQUENCY_MHZ   (int)(HAL_RCC_GetHCLKFreq()/1000000)		// 自动获取STM32时钟主频
	
void DHT11_Delay_us(__IO uint32_t delay)  
{
    int last, curr, val;
    int temp;

    while (delay != 0)
    {
        temp = delay > 900 ? 900 : delay;
        last = SysTick->VAL;
        curr = last - CPU_FREQUENCY_MHZ * temp;
        if (curr >= 0)
        {
            do
            {
                val = SysTick->VAL;
            }
            while ((val < last) && (val >= curr));
        }
        else
        {
            curr += CPU_FREQUENCY_MHZ * 1000;
            do
            {
                val = SysTick->VAL;
            }
            while ((val <= last) || (val > curr));
        }
        delay -= temp;
    }
}

(二)dht11.h 代码

cpp 复制代码
#include "main.h"

#ifndef __DHT11_H_
#define __DHT11_H_
/**
 *
 * 如果未用CubeMX配置引脚,可以将下面代码的注释取消,并替换后面的GPIOB以及GPIO_PIN_1 
 * 例如: 使用了PA5引脚,则应将 GPIOB 替换成 GPIOA ,将 GPIO_PIN_1 替换成 GPIO_PIN_5
 *
***/

// #define DHT11_GPIO_Port GPIOB
// #define DHT11_Pin       GPIO_PIN_1

#define DHT11_IO_Read   HAL_GPIO_ReadPin(DHT11_GPIO_Port,DHT11_Pin)                   //读DHT11引脚电平
#define DHT11_IO_SET    HAL_GPIO_WritePin(DHT11_GPIO_Port,DHT11_Pin,GPIO_PIN_SET)     //DHT11引脚置高电平
#define DHT11_IO_RESET  HAL_GPIO_WritePin(DHT11_GPIO_Port,DHT11_Pin,GPIO_PIN_RESET)   //DHT11引脚置低电平
  
#define INPUT 0   //引脚输入模式
#define OUTPUT 1  //引脚输出模式

void DHT11_Delay_us(__IO uint32_t delay);           // 微妙级延时函数
void DHT11_PIN_Mode(int Mode);  										// 引脚模式配置函数
void DHT11_Start(void);         										// 起始信号发送函数
uint8_t DHT11_Check(void);      										// DHT11应答检测函数
uint8_t DHT11_Read_Bit(void);   										// 读取一个数据位(bit),8 bit = 1 byte
uint8_t DHT11_Read_Byte(void);  										// 读取一个字节的数据
uint8_t DHT11_READ_DATA(float *temp, float *humi);  // 温湿度数据读取函数

#endif

(三)main.c 中调用

注意: 在main.c中需要包含dht11.h、stdio.h两个头文件,声明两个浮点变量和一个串口发送缓冲数组。使用的时候不需要初始化,直接放while里面循环读取就可,但必须要加上延时。

cpp 复制代码
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
#include "usart.h"
#include "gpio.h"

/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
#include "dht11.h"
#include "stdio.h"
/* USER CODE END Includes */

/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
float Humi,Temp;
char  DHT11_TX[40];
/* USER CODE END PTD */

/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */

/* USER CODE END PD */

/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */

/* USER CODE END PM */

/* Private variables ---------------------------------------------------------*/

/* USER CODE BEGIN PV */

/* USER CODE END PV */

/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
/* USER CODE BEGIN PFP */

/* USER CODE END PFP */

/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */

/* USER CODE END 0 */

/**
  * @brief  The application entry point.
  * @retval int
  */
int main(void)
{

  /* USER CODE BEGIN 1 */

  /* USER CODE END 1 */

  /* MCU Configuration--------------------------------------------------------*/

  /* Reset of all peripherals, Initializes the Flash interface and the Systick. */
  HAL_Init();

  /* USER CODE BEGIN Init */

  /* USER CODE END Init */

  /* Configure the system clock */
  SystemClock_Config();

  /* USER CODE BEGIN SysInit */

  /* USER CODE END SysInit */

  /* Initialize all configured peripherals */
  MX_GPIO_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */

  /* USER CODE END 2 */

  /* Infinite loop */
  /* USER CODE BEGIN WHILE */
  while (1)
  {
		HAL_Delay(2000);
		DHT11_READ_DATA(&Temp,&Humi);
		sprintf(DHT11_TX,"温度:%0.1f 度	湿度:%0.1f %%\r\n",Temp,Humi);
		HAL_UART_Transmit(&huart1,(uint8_t*)DHT11_TX,40,200);
    /* USER CODE END WHILE */

    /* USER CODE BEGIN 3 */
  }
  /* USER CODE END 3 */
}

五、效果展示

相关推荐
怪小庄吖1 小时前
翻译:How do I reset my FPGA?
经验分享·嵌入式硬件·fpga开发·硬件架构·硬件工程·信息与通信·信号处理
雯宝8 小时前
STM32 GPIO工作模式
stm32·单片机·嵌入式硬件
辰哥单片机设计9 小时前
STM32项目分享:智能厨房安全检测系统
stm32·单片机·嵌入式硬件
lshzdq11 小时前
【嵌入式开发】stm32 st-link 烧录
嵌入式硬件
山羊硬件Time12 小时前
详解单片机学的是什么?(电子硬件)
单片机·硬件工程师·硬件开发·电子工程师·电子硬件
Chambor_mak12 小时前
stm32单片机个人学习笔记14(USART串口数据包)
stm32·单片机·学习
tadus_zeng12 小时前
51单片机(三) UART协议与串口通信实验
单片机·嵌入式硬件·51单片机
ZLG_zhiyuan13 小时前
ZLG嵌入式笔记 | 电源设计避坑(下)
单片机·嵌入式硬件
wenchm14 小时前
细说STM32F407单片机电源低功耗StopMode模式及应用示例
stm32·单片机·嵌入式硬件
7yewh15 小时前
嵌入式知识点总结 C/C++ 专题提升(七)-位操作
c语言·c++·stm32·单片机·mcu·物联网·位操作