文章标题
- 一、DHT11简介
- 二、数据手册分析
- 2.1 接口说明
- 2.2 串行通信说明
- 2.2.1 单总线通信
- 2.2.2 单总线传输数据位定义
- 2.2.3 时序图
- 三、DHT11程序设计
- 3.1 初始化GPIO
- 3.2 发送起始信号
- 3.3 接收一个字节数据
- 3.4 接收温湿度信息并校准
- 四、总结
一、DHT11简介
DHT11是一款常用的数字温湿度传感器。传感器包括一个电容式感湿元件和一个 NTC 测温元件,能够测量皱纹环境的温湿度,常用于暖通空调、除湿器、农业、冷链仓储等方面。
二、数据手册分析
2.1 接口说明
2.2 串行通信说明
DHT11通过串行通信的方式,将采集到的环境温湿度信息传递给单片机。数据手册中,针对DHT11的串行通信做了详细介绍。
2.2.1 单总线通信
DHT11 器件采用简化的单总线通信。单总线即只有一根数据线,系统中的数据交换、控制均由单总线完成。设备(主机或从机)通过一个漏极开路或三态端口连至数据线,以允许设备在不发送数据时能够释放总线,而让其它设备使用总线;单总线通常要求外接一个约 4.7kΩ 的上拉电阻,这样,当总线闲置时,其状态为高电平。由于它们是主从结构,只有主机呼叫从机时,从机才能应答,因此主机访问器件都必须严格遵循单总线序列,如果出现序列混乱,器件将不响应主机。
2.2.2 单总线传输数据位定义
DHT11的DATA引脚,用于单片机与 DHT11 之间的通讯和同步,采用单总线数据格式,一次传送 40 位数据,高位先出。数据格式
8bit 湿度整数数据 + 8bit 湿度小数数据 + 8bit 温度整数数据 + 8bit 温度小数数据 + 8bit 校验位
数据手册中写明,湿度的小数部分为0。8bit 校验位等于所得结果的末 8 位。
对于校验位,数据手册中举例说明。比如接收到的40位数据为
0011 0101 | 0000 0000 | 0001 1000 | 0000 0100 | 0101 0001 |
---|---|---|---|---|
湿度高8位 | 湿度低8位 | 温度高8位 | 温度低8位 | 校验位 |
计算 0011 0101 + 0000 0000 + 0001 1000 + 0000 0100 + 0101 0001 = 0101 0001,与接收到的校验位相等,校验通过。如果校验不通过,则将此次接收到的数据丢弃,重新接收数据。
2.2.3 时序图
根据上面的介绍,如果单片机想要读取数据,需要先发送一个起始信号。起始信号需要拉低数据线至少18ms。
DHT11检测到起始信号之后,等待起始信号低电平结束,然后输出应答信号。应答信号是先将数据线拉低83us,再拉高87us。
然后DHT11就开始输出数据了,“0”和“1”的时序图如下
总时序图如下
三、DHT11程序设计
3.1 初始化GPIO
根据上面的介绍,STM32的GPIO既需要用作输出,也需要用作输入。因此,STM32的GPIO需要有两种配置
/**==============================================================================*函数名称:Drv_Dht11_Gpio_OutInSet*函数功能:DHT11引脚输出/输入设置*输入参数:state:OUT:输出(0);IN:输入(1)*返回值:无*备 注:无*==============================================================================*/
void Drv_Dht11_Gpio_OutInSet (u8 state)
{// 结构体定义GPIO_InitTypeDef GPIO_InitStructure;// 开启时钟RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA, ENABLE);// 初始化GPIO结构体GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2;if (state){GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IN_FLOATING; // 浮空输入}else{GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP; // 推挽式输出}GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;GPIO_Init(GPIOA,&GPIO_InitStructure);
}
.h文件添加下面程序
// GPIO模式
#define OUT 0 // 输出模式
#define IN 1 // 输入模式
3.2 发送起始信号
单片机的 I/O 设置为输出同时输出低电平,且低电平保持时间不能小于 18ms(最大不得超过 30ms),然后单片机的 I/O 设置为输入状态,由于上拉电阻,单片机器的 I/O 即 DHT11 的 DATA 数据线也随之变高,等待 DHT11 作出回答信号。程序设计如下
/**==============================================================================*函数名称:Drv_Dht11_Start*函数功能:向DHT11发送起始信号*输入参数:state:OUT:输出(0);IN:输入(1)*返回值:无*备 注:无*==============================================================================*/
void Drv_Dht11_Start (void)
{OLED_SDA_Set(); // 拉高一小段时间delay_us(30);Drv_Dht11_Gpio_OutInSet(OUT); // GPIO配置为输出模式OLED_SDA_Clr(); // 拉低数据线delay_ms(20); // 保持20msOLED_SDA_Set(); // 拉高一小段时间delay_us(30);Drv_Dht11_Gpio_OutInSet(IN); // GPIO配置为输入模式
}
3.3 接收一个字节数据
/**==============================================================================*函数名称:Med_Dht11_ReceOneByte*函数功能:接收一帧数据*输入参数:无*返回值:一字节接收数据*备 注:无*==============================================================================*/
u8 Med_Dht11_ReceOneByte (void)
{u8 tempVar = 0; // 临时循环变量u8 receData = 0; // 接收数据for (tempVar = 0;tempVar < 8;tempVar ++){while (!DHT11_SDA_DATA); // 等待54us的低电平过去delay_us(30); // 延时30us之后判断是0还是1// 如果30us之后依旧为高电平if (DHT11_SDA_DATA){receData |= 1; // 接收数据为1}while (DHT11_SDA_DATA); // 等待高电平过去receData <<= 1; // 左移}return receData;
}
3.4 接收温湿度信息并校准
/**==============================================================================*函数名称:App_Dht11_Result_Process*函数功能:处理接收结果,得出正确的温湿度信息*输入参数:无*返回值:无*备 注:无*==============================================================================*/
u8 receData[4]; // 存储温湿度结果
void App_Dht11_Result_Process (void)
{u8 receCheck = 0; // 接收到的校验码u8 tempCal = 0; // 临时计算变量Drv_Dht11_Start(); // 发送一个起始信号DHT11_SDA_Set(); // 拉高SDA// 等待应答信号if (!DHT11_SDA_DATA){while (!DHT11_SDA_DATA); // 等待83us低电平结束while (DHT11_SDA_DATA); // 等待87us高电平结束// 开始接收数据receData[0] = Med_Dht11_ReceOneByte(); // 湿度整数receData[1] = Med_Dht11_ReceOneByte(); // 湿度小数receData[2] = Med_Dht11_ReceOneByte(); // 温度整数receData[3] = Med_Dht11_ReceOneByte(); // 温度小数receCheck = Med_Dht11_ReceOneByte(); // 校验码tempCal = receData[0] + receData[1] + receData[2] + receData[3];// 如果校验失败if (tempCal != receCheck){// 清空接收receData[0] = receData[1] = receData[2] = receData[3] = 0;}}
}
四、总结
实际上面的程序设计有一些不足,比如某些地方不需要再拉高SDA线,在等到时使用了while语句但是没有超时检测。但是由于博主的DHT11坏了,目前买的新的还没到,无法继续调试,因此这里说明一下。后续会修改完善程序,补充应用实例。——2023年6月26日