STM32+DHT11读取温湿度

发布时间 2023-03-22 21:16:52作者: 浅晓寒

STM32+DHT11读取温湿度

使用的是stm32f103c8t6的最小系统开发版。

DHT11对时序要求很高,要注意延时函数的延时时间是否准确,延时函数使用的是正点原子的延时函数。通讯过程如图:

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

RCC_APB2PeriphClockCmd (RCC_APB2Periph_GPIOB,ENABLE);
GPIO_InitTypeDef GPIO_InitStructure;
GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;	//复用推挽输出
GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
GPIO_Init (GPIOB,&GPIO_InitStructure);             

//查DHT11数据手册
GPIO_ResetBits(GPIOB,GPIO_Pin_12);	//主机线拉低20ms
delay_ms(20); 
GPIO_SetBits(GPIOB,GPIO_Pin_12);	//主机线拉高25us
delay_us(25);                                                    

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

数字0信号:

数字1信号:

GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;	//数据线设为上拉输入,准备外设读取
GPIO_Init (GPIOB,&GPIO_InitStructure);              

//DHT11响应
dhti=0;
while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==0)		//等待80us低电平结束
{dhti++;if(dhti>10000) {break;}}						//如果满足条件则单片机与DHT11通讯失败
dhti=0;
while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==1)		//等待80us高电平结束
{dhti++;if(dhti>10000) {break;}} //DHT11

for(char j=0;j<5;j++)	//开始读取40位数据:32位有效数据位+8位校验位
{
    for(char i=0;i<8;i++)
    {
        dhti=0;
        while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==0)	//等待50us低电平结束
        {dhti++;if(dhti>10000) {break;}}
        delay_us(30);	//延时30us后如果还是高电平则是数字1信号,否则是0信号
        Data[j] <<=1;
        Data[j] |= (unsigned char)GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12);	//读取PB12的数据
        dhti=0;
        while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==1)	//等待高电平结束,准备下次数据的读取
        {dhti++;if(dhti>10000) {break;}};
    }
}

DHT11.c

#include "stm32f10x.h"                  // Device header
#include "Delay.h"
#include <string.h>
unsigned char Data[5];

uint16_t dhti;

/*
	DHT11接线:out--PB12
*/

char DHT11_GetData(void)
{
	memset(Data,0,5);
	RCC_APB2PeriphClockCmd (RCC_APB2Periph_GPIOB,ENABLE);
	GPIO_InitTypeDef GPIO_InitStructure;
	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;	//数据线设为复用推挽输出,主机准备与DHT11通信
	GPIO_InitStructure.GPIO_Pin = GPIO_Pin_12;
	GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
	GPIO_Init (GPIOB,&GPIO_InitStructure);             
	
	//查DHT11数据手册
	GPIO_ResetBits(GPIOB,GPIO_Pin_12);	//主机线拉低20ms
	delay_ms(20); 
	GPIO_SetBits(GPIOB,GPIO_Pin_12);	//主机线拉高20-40us
	delay_us(25);                           

	GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;	//数据线设为上拉输入,准备外设读取
	GPIO_Init (GPIOB,&GPIO_InitStructure);              
	
	dhti=0;
	while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==0)		//等待80us低电平结束
	{dhti++;if(dhti>10000) {break;}}
	dhti=0;
	while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==1)		//等待80us高电平结束
		{dhti++;if(dhti>10000) {break;}} //DHT11
	
	for(char j=0;j<5;j++)	//开始读取40位数据:32位有效数据位+8位校验位
	{
		for(char i=0;i<8;i++)
		{
			dhti=0;
			while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==0)	//等待50us低电平结束
				{dhti++;if(dhti>10000) {break;}}
			delay_us(30);	//延时30us后如果还是高电平则是数字1信号,否则是0信号
			Data[j] <<=1;
			Data[j] |= (unsigned char)GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12);	//读取PB12的数据
			dhti=0;
			while(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_12)==1)	//等待高电平结束,准备下次数据的读取
				{dhti++;if(dhti>10000) {break;}};
		}
	}
	if(Data[4]==(char)(Data[0]+Data[1]+Data[2]+Data[3]))	//校验位
		return 1;
	else
		return 0;
}
主函数调用DHT11_GetData()并声明extern unsigned char Data[5]即可读取温湿度。
数据格式:8bit湿度整数数据+8bit湿度小数数据+8bit温度整数数据+8bit温度小数数据+8bit校验位