1.DHT11简介

DHT11是一款温湿度一体化的数字传感器,该传感器包括一个电阻式测湿元件和一个NTC测温元件,并与一个高性能8位单片机连接。通过单片机微处理器简单的电路连接就能够实时的采集湿度和温度。DHT11与STM32之间能采用简单的单总线进行通讯,仅需一个IO口。传感器内部温度和湿度数据40个bit的数据一次性传给单片机,数据采用校验和方式进行校验,有效的保证数据传输的准确性。 DHT11采用单总线数据格式,即单个数据引脚端口完成输入输出双向传输。其数据包由5个字节(40Bit)组成。数据分小数和整数部分,一次完整的数据传输为40bit,高位先出。数据格式如下图示 根据以上数据,即可算出温度和湿度的值,计算方法如下:

湿度 = byte4.byte3
温度 = byte2.byte1
校验 = byte4 + byte3 + byte2 + byte1

DHT11和单片机的一次通讯最大为3ms左右,建议主机连续读取时间间隔不要小于100ms,下面是DHT11的三种时序: ⏩ 数据发送时序 首先主机发送开始信号,即拉低数据线保持t1(至少18ms)时间,接着拉高数据线t2(20 ~ 40us)时间;然后读取DHT11的响应,正常的话DHT11会拉低数据线并保持t3(40 ~ 50us)时间作为响应信号,接着DHT11拉高数据线并保持t4(40 ~ 50us)时间后,开始传输数据 ⏩ DHT11输出0时序 ⏩ DHT11输出1时序

2.硬件设计

D1指示灯用来提示系统运行状态,DS18B20温度传感器用来检测环境温度,串口1用来打印温度值

* 指示灯D1
* USART1串口
* PG11
* TIM7(提供us延时)

3.软件设计

3.1 STM32CubeMX设置 ➡️ RCC设置外接HSE,时钟设置为72M ➡️ PC0设置为GPIO推挽输出模式、上拉、高速、默认输出电平为高电平 ➡️ USART1选择为异步通讯方式,波特率设置为115200Bits/s,传输数据长度为8Bit,无奇偶校验,1位停止位 ➡️ PG11设置为GPIO推挽输出模式、上拉、高速 ➡️ 激活TIM7,预分频因子设为72-1,向上计数,自动重载值为65535;因此计数器CNT_CLK = 1MHz,计数器周期为1us ➡️输入工程名,选择路径(不要有中文),选择MDK-ARM V5;勾选Generated periphera initialization as a pair of ‘.c/.h’ files per IP ;点击GENERATE CODE,生成工程代码

3.2 MDK-ARM软件编程 ➡️ 在tim.c文件下实现微秒延时(us)函数

void delay_us(uint16_t us){
  uint16_t differ = 0xffff-us-5;
  //设定TIM7计数器起始值				
  __HAL_TIM_SET_COUNTER(&htim7,differ);
  //启动定时器	
  HAL_TIM_Base_Start(&htim7);			
	
  while(differ < 0xffff-5){	//判断
    //查询计数器的计数值
    differ = __HAL_TIM_GET_COUNTER(&htim7);		
  }
  HAL_TIM_Base_Stop(&htim7);
}

➡️ 创建DHT11温湿度传感器驱动文件DHT11.c 和相关头文件DHT11.h

void DHT11_IO_IN(void){  //IO口方向设置为输入
  GPIO_InitTypeDef GPIO_InitStructure;
  GPIO_InitStructure.Pin = GPIO_PIN_11;
  GPIO_InitStructure.Mode = GPIO_MODE_INPUT;
  HAL_GPIO_Init(GPIOG,&GPIO_InitStructure);
}

void DHT11_IO_OUT(void){  //IO口方向设置为输出
  GPIO_InitTypeDef GPIO_InitStructure;
  GPIO_InitStructure.Pin = GPIO_PIN_11;
  GPIO_InitStructure.Mode = GPIO_MODE_OUTPUT_PP;
  GPIO_InitStructure.Speed = GPIO_SPEED_FREQ_HIGH;
  HAL_GPIO_Init(GPIOG,&GPIO_InitStructure);
}


void DHT11_Rst(void){                 
  DHT11_IO_OUT(); 	//设置为输出
  DHT11_DQ_OUT_LOW; 	//拉低
  HAL_Delay(20);    	//至少18ms
  DHT11_DQ_OUT_HIGH; 	//拉高 
  delay_us(30);     	//至少20~40us
}

uint8_t DHT11_Check(void){   
  uint8_t retry=0;
  DHT11_IO_IN();      
  while (DHT11_DQ_IN&&retry<100){ //拉低40~80us
    retry++;
    delay_us(1);
  };	 
  if(retry>=100)return 1;
  else retry=0;
  while (!DHT11_DQ_IN&&retry<100){ //拉高40~80us
    retry++;
    delay_us(1);
  };
  if(retry>=100)return 1;	    
  return 0;	//检测到DHT11返回0
}

uint8_t DHT11_Read_Bit(void){
  uint8_t 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;		   
}

uint8_t DHT11_Read_Byte(void){        
  uint8_t i,dat;
  dat=0;
  for (i=0;i<8;i++){
    dat<<=1; 
    dat|=DHT11_Read_Bit();
  }						    
  return dat;
}

uint8_t DHT11_Read_Data(uint16_t *temp,uint16_t *humi){        
  uint8_t buf[5];
  uint8_t i;
  DHT11_Rst();
  if(DHT11_Check()==0){
    for(i=0;i<5;i++){
      buf[i]=DHT11_Read_Byte();
    }
    if((buf[0]+buf[1]+buf[2]+buf[3])==buf[4]){
      *humi=(buf[0]<<8) + buf[1];
      *temp=(buf[2]<<8) + buf[3];
    }
  }else return 1;
  return 0;	    
}
  	 
uint8_t DHT11_Init(void){
  //PG11的初始化已经在cubemx中完成,可以忽略此段初始化代码
  GPIO_InitTypeDef GPIO_Initure;	
  __HAL_RCC_GPIOG_CLK_ENABLE();			
  GPIO_Initure.Pin=GPIO_PIN_11;           
  GPIO_Initure.Mode=GPIO_MODE_OUTPUT_PP;  
  GPIO_Initure.Pull=GPIO_PULLUP;         
  GPIO_Initure.Speed=GPIO_SPEED_FREQ_HIGH;
  HAL_GPIO_Init(GPIOG,&GPIO_Initure);    
  DHT11_Rst();
  return DHT11_Check();
}

➡️ 在main.c文件下编写DHT11测试代码

int main(void){
  /* USER CODE BEGIN 1 */
  uint16_t temperature;
  uint16_t humidity;
  /* USER CODE END 1 */
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_TIM7_Init();
  MX_USART1_UART_Init();
  /* USER CODE BEGIN 2 */
  while(DHT11_Init()){
    printf("DHT11 Checked failed!!!\r\n");
    HAL_Delay(500);
  }
  printf("DHT11 Checked Sucess!!!\r\n");
  /* USER CODE END 2 */
  while (1){
    DHT11_Read_Data(&temperature,&humidity);
    printf("DHT11 Temperature = %d.%d degree\r\n",temperature>>8,temperature&0xff);
    printf("DHT11 Humidity = %d.%d%%\r\n",humidity>>8,humidity&0xff);		
    HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_0);
    HAL_Delay(500);
  }
}

4.下载验证

图片编译无误下载到开发板后,可以看到D1指示灯不断闪烁,串口不断打印出当前温湿度值。串口输出如下信息:

  • DHT11 Checked Sucess!!!
  • DHT11 Temperature = 28.20 degree
  • DHT11 Humidity = 30.21%