STM32与DHT11温湿度传感器实战教程:代码详解

DHT11简介


DHT11的供电电压为 3-5.5V。
传感器上电后,要等待 1s 以越过不稳定状态在此期间无需发送任何指令。
电源引脚(VDD,GND)之间可增加一个100nF 的电容,用以去耦滤波。
DATA 用于微处理器与DHT11之间的通讯和同步,采用单总线数据格式,一次通讯时间4ms左右,数据分小数部分和整数部分,当前小数部分用于以后扩展,现读出为零。
操作流程如下:
一次完整的数据传输为40bit,高位先出。
数据格式:8bit湿度整数数据 + 8bit湿度小数数据
+8bi温度整数数据 + 8bit温度小数数据
+8bit校验和
数据传送正确时校验和数据等于“8bit湿度整数数据+8bit湿度小数数据
+8bit温度整数数据+8bit温度小数数据”所得结果的末8位

通讯过程(单线双向)


总线空闲状态为高电平,主机把总线拉低等待DHT11响应,主机把总线拉低必须大于18毫秒,保证DHT11能检测到起始信号。DHT11接收到主机的开始信号后,等待主机开始信号结束,然后发送80us低电平响应信号。
主机发送开始信号结束后,延时等待20-40us后,读取DHT11的响应信号,主机发送开始信号后,可以切换到输入模式,或者输出高电平均可, 总线由上拉电阻拉高。

总线为低电平(80us低电平响应信号),说明DHT11发送响应信号,DHT11发送响应信号后,再把总线拉高80us,准备发送数据。
每1bit数据都以50us低电平时隙开始,高电平的长短定了数据位是0还是1。
格式见下面图示:
数字0信号表示方法

数字1信号表示方法

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

#ifndef __DHT11_H
#define __DHT11_H
#include "stm32f10x.h"

// 位带操作宏定义
#define BITBAND(addr, bitnum) ((addr & 0xF0000000) + 0x2000000 + ((addr & 0xFFFFF) << 5) + (bitnum << 2))
#define MEM_ADDR(addr)  *((volatile unsigned long  *)(addr))
#define BIT_ADDR(addr, bitnum)   MEM_ADDR(BITBAND(addr, bitnum))

// GPIOA的ODR和IDR寄存器地址
#define GPIOA_ODR_ADDR    (GPIOA_BASE + 12) // GPIOA的输出数据寄存器地址,0x4001080C 
#define GPIOA_IDR_ADDR    (GPIOA_BASE + 8)  // GPIOA的输入数据寄存器地址,0x40010808

// 定义PBout和PBin宏
#define PAout(n)   BIT_ADDR(GPIOA_ODR_ADDR, n)  //输出 
#define PAin(n)    BIT_ADDR(GPIOA_IDR_ADDR, n)  //输入 

/************************************************************************/

#define	DHT11_DQ_OUT PAout(15) //数据端口	PA15 
#define	DHT11_DQ_IN  PAin(15)  //数据端口	PA15

#define DHT11_GPIO_PORT    	GPIOA			            //GPIO端口
#define DHT11_GPIO_CLK 	    RCC_APB2Periph_GPIOA		//GPIO端口时钟
#define DHT11_GPIO_PIN		GPIO_Pin_15			        //连接到SCL时钟线的GPIO

/************************************************************************/

void DHT11_Rst(void);
uint8_t DHT11_Check(void);
uint8_t DHT11_Read_Bit(void);
uint8_t DHT11_Read_Byte(void);
uint8_t DHT11_Read_Data(uint8_t *temp,uint8_t *humi);	 
uint8_t DHT11_Init(void);

#endif

#include "stm32f10x.h"                  // Device header
#include "DHT11.h"
#include "Delay.h"

/*DHT11数据线为输出模式*/
void DHT11_IO_OUT(void)
{
	RCC_APB2PeriphClockCmd(DHT11_GPIO_CLK | RCC_APB2Periph_AFIO, ENABLE); //使能GPIO时钟和AFIO时钟
	GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE); //禁用JTAG,保留SWD
	
	GPIO_InitTypeDef  GPIO_InitStructure;	
 	GPIO_InitStructure.GPIO_Pin = DHT11_GPIO_PIN;			
 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; //推挽输出模式	 
 	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
 	GPIO_Init(DHT11_GPIO_PORT, &GPIO_InitStructure);				
}

/*DHT11数据线为输入模式*/
void DHT11_IO_IN(void)
{
	RCC_APB2PeriphClockCmd(DHT11_GPIO_CLK | RCC_APB2Periph_AFIO, ENABLE); //使能GPIO时钟和AFIO时钟
	GPIO_PinRemapConfig(GPIO_Remap_SWJ_JTAGDisable, ENABLE); //禁用JTAG,保留SWD
	
	GPIO_InitTypeDef  GPIO_InitStructure;	
 	GPIO_InitStructure.GPIO_Pin = DHT11_GPIO_PIN;
 	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; //浮空输入模式
 	GPIO_Init(DHT11_GPIO_PORT, &GPIO_InitStructure);
}

/*复位DHT11传感器*/
void DHT11_Rst(void)	   
{                 
	DHT11_IO_OUT(); 	//设置输出模式
	DHT11_DQ_OUT = 0; 	//将数据线拉低,等待DHT11响应
	Delay_ms(20);    	//拉低至少18ms,保证DHT11能检测到起始信号
	DHT11_DQ_OUT = 1; 	//将数据线拉高 
	Delay_us(30);     	//主机拉高20~40us,读取DHT11的响应信号
}

/*检查DHT11是否存在*/
//返回1:未检测到DHT11的存在
//返回0:检测到DHT11的存在
uint8_t DHT11_Check(void) 	   
{   
	uint8_t retry = 0;
	DHT11_IO_IN(); //设置输入模式 
	//DHT11接收到主机的开始信号后,等待主机开始信号结束,然后发送80us低电平响应信号
	//DHT11发送响应信号后,再把总线拉高80us,准备发送数据
    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读取一个位*/
//返回值:数据位1或0
uint8_t DHT11_Read_Bit(void)
{
 	uint8_t retry = 0;
	//每1bit数据都以50us低电平时隙开始
	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
	//高电平的长短定了数据位是0还是1
	//26us-28us表示0,70us表示1
	if(DHT11_DQ_IN) return 1;
	else return 0;
}

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

/*DHT11读取一次数据*/
//temp:温度值(范围:0~50°)
//humi:湿度值(范围:20%~90%)
//返回值:0,正常;1,读取失败
uint8_t DHT11_Read_Data(uint8_t *temp,uint8_t *humi)    
{        
 	uint8_t buf[5];
	uint8_t i;
	DHT11_Rst();
	if(DHT11_Check() == 0)
	{
		for(i = 0; i < 5; i++)	//读取40位数据
		{
			buf[i] = DHT11_Read_Byte();
		}
		//数据格式:8bit湿度整数数据 + 8bit湿度小数数据 + 8bi温度整数数据 + 8bit温度小数数据 + 8bit校验和
		//8bit校验和 = 8bit湿度整数数据 + 8bit湿度小数数据 + 8bit温度整数数据 + 8bit温度小数数据
		if((buf[0]+buf[1]+buf[2]+buf[3])==buf[4]) //校验数据
		{
			*humi = buf[0];
			*temp = buf[2];
		}
	}else return 1;
	return 0;
}

//初始化DHT11的IO口同时检测DHT11是否存在
//返回1:不存在
//返回0:存在    	 
uint8_t DHT11_Init(void)
{	    
	DHT11_Rst();  //复位DHT11
	return DHT11_Check();	//等待DHT11的回应
} 

作者:七点半.

物联沃分享整理
物联沃-IOTWORD物联网 » STM32与DHT11温湿度传感器实战教程:代码详解

发表回复