本篇介绍STM32如何外接温湿度传感器实现当前环境温湿度的读取,并显示到OLED屏幕上。
1 DTH11温湿度传感器
DHT11数字温湿度传感器是一款含有已校准数字信号输出的温湿度复合传感器,包括一个电阻式感湿元件和一个NTC测温元件。
1.1 数据读取协议
微控制器MCU与 DHT11之间的通讯和同步,采用单总线数据格式,一次通讯时间4ms左右。
用户MCU发送一次开始信号后,DHT11从低功耗模式转换到高速模式,等待主机开始信号结束后,DHT11发送响应信号,送出40bit的数据,并触发一次信号采集,用户可选择读取部分数据。
从模式下,DHT11接收到开始信号触发一次温湿度采集,如果没有接收到主机发送开始信号,DHT11不会主动进行温湿度采集。采集数据后转换到低速模式。
1.1.1 起始信号
总线空闲状态为高电平,MCU把总线拉低等待DHT11响应,MCU把总线拉低必须大于18毫秒,保证DHT11能检测到起始信号。
DHT11接收到主机的开始信号后,等待MCU的开始信号结束,然后发送80us低电平响应信号。
MCU发送开始信号结束后,延时等待20-40us后,读取DHT11的响应信号,MCU发送开始信号后,可以切换到输入模式,或者输出高电平均可,总线由上拉电阻拉高。
1.1.2 数据数字信号
总线为低电平,说明DHT11发送响应信号,DHT11发送响应信号后,再把总线拉高80us,准备发送数据,每一bit数据都以50us低电平时隙开始,高电平的长短定了数据位是0还是1。
数字0和数字1的表示,如下面图示:
- 数字0:50us低电平开始后,26-28us的高电平表示0
- 数字1:50us低电平开始后,70us的高电平表示1
如果读取响应信号为高电平,则DHT11没有响应,需要检查线路是否连接正常。
当最后一bit数据传送完毕后,DHT11拉低总线50us,随后总线由上拉电阻拉高进入空闲状态。
1.1.3 温湿度数据格式
一次完整的数据传输为40bit,高位先出。数据分小数部分和整数部分,数据格式:
- 8bit湿度整数数据
- 8bit湿度小数数据
- 8bit温度整数数据
- 8bit温度小数数据
- 8bit校验和
数据传送正确时校验和数据等于“ 8bit 湿度整数数据 +8bit 湿度小数数据+8bit温度整数数据 +8bit 温度小数数据 ”所得结果的末8位。
1.2 硬件接线
DHT11的数据读取只需要一根线,我使用的是PB8,另外,OLED用来显示温湿度的值,使用IIC通信,使用的是PB6和PB7。
2 程序编写
根据DHT11的数据读取协议,编写对应的数据读取函数。
2.1 DHT11复位和检测响应函数
首先是MCU向DHT11发送的起始信号,拉低20ms,再拉高30us。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34
| u8 DHT11RstAndCheck(void) { u8 timer = 0;
__set_PRIMASK(1); DHT11_OUT = 0; delay_ms(20); DHT11_OUT = 1; delay_us(30); while (!DHT11_IN) { timer++; delay_us(1); } if (timer>100 || timer<20) { __set_PRIMASK(0); return 0; } timer = 0; while (DHT11_IN) { timer++; delay_us(1); } __set_PRIMASK(0); if (timer>100 || timer<20) { return 0; } return 1; }
|
2.2 数据读取
MCU向DHT11发送起始信号后,就可以接收DHT11的数据返回了,一次读取湿度和温度即可。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62
| u8 DHT11ReadByte(void) { u8 i; u8 byt = 0;
__set_PRIMASK(1); for (i=0; i<8; i++) { while (DHT11_IN); while (!DHT11_IN); delay_us(40); byt <<= 1; if (DHT11_IN) { byt |= 0x01; } } __set_PRIMASK(0);
return byt; }
u8 DHT11ReadData(float *Humi, float *Temp) { s8 sta = 0; u8 i; u8 buf[5];
if (DHT11RstAndCheck()) { for(i=0;i<5;i++) { buf[i]=DHT11ReadByte(); } if(buf[0]+buf[1]+buf[2]+buf[3] == buf[4]) { u8 H_inte = buf[0]; u8 H_frac = buf[1]; u8 T_inte = buf[2]; u8 T_frac = buf[3]; char tmp1[8], tmp2[8]; sprintf(tmp1, "%d.%d",H_inte,H_frac); sscanf(tmp1, "%f", Humi); sprintf(tmp2, "%d.%d",T_inte,T_frac); sscanf(tmp2, "%f", Temp); } sta = 0; } else { *Humi = 88; *Temp = 88; sta = 1; } return sta; }
|
2.3 初始化
使用DHT11之前,进行引脚的初始化和器件的初始化。
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| u8 DHT11Init(void) { GPIO_InitTypeDef GPIO_InitStructure;
RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOC,ENABLE); GPIO_SetBits(GPIOB,GPIO_Pin_8); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_OD; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOB, &GPIO_InitStructure);
return DHT11RstAndCheck(); }
|
3 测试
在移植过U8g2库的hello_world例程上进行修改,在屏幕上显示温湿度。注意摄氏度单位的小圆圈,不知道怎么直接以符号的形式显示出来,我这里是单独画了一个小空心圆。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| int main(void) { delay_init(); LED_Init(); IIC_Init(); u8g2_t u8g2; u8g2Init(&u8g2); u8g2_SetFontMode(&u8g2, 1); u8g2_SetFont(&u8g2, u8g2_font_unifont_t_symbols); DHT11Init(); float Temp = 0; float Humi = 0; char strTemp[32]; char strHumi[32]; while(1) { u8g2_FirstPage(&u8g2); do { DHT11ReadData(&Humi, &Temp); sprintf(strTemp, "Temp: %.1f C", Temp); sprintf(strHumi, "Humi: %.1f %%", Humi); u8g2_ClearBuffer(&u8g2); u8g2_DrawStr(&u8g2, 0, 30, strTemp); u8g2_DrawCircle(&u8g2, 84, 22, 2, U8G2_DRAW_ALL); u8g2_DrawStr(&u8g2, 0, 60, strHumi); u8g2_SendBuffer(&u8g2); delay_ms(3000); } while (u8g2_NextPage(&u8g2)); } }
|
测试效果如下:
4 总结
本篇介绍了如何在STM32上外接温湿度DHT11实现温湿度数据的读取,并通过OLED进行数据显示。