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,保留SWDGPIO_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,保留SWDGPIO_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表示1if(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(); //复位DHT11return DHT11_Check(); //等待DHT11的回应
}