STM32与DHT11打造智能温湿度监测系统从硬件搭建到功能扩展全指南在物联网和智能家居快速发展的今天环境监测已成为许多项目的核心需求。本文将带你从零开始构建一个基于STM32和DHT11的智能温湿度监测系统不仅实现基础数据采集还包含OLED显示、数据记录和按键交互等进阶功能。1. 项目规划与硬件选型1.1 核心组件介绍STM32F103C8T6蓝莓派开发板作为主控制器具有以下优势72MHz Cortex-M3内核性能充沛64KB Flash 20KB SRAM丰富的外设接口GPIO、I2C、USART等成本低廉且社区资源丰富DHT11温湿度传感器特点数字信号输出免去ADC转换湿度测量范围20-90%RH (±5%)温度测量范围0-50℃ (±2℃)单总线通信接线简单OLED显示屏选择考量0.96寸I2C接口SSD1306模块128x64分辨率自发光、高对比度相比LCD1602显示信息更丰富1.2 硬件连接示意图STM32F103C8T6 外设模块 ---------- ---------- | | | | | PB5 |-----| DHT11 | | | | DATA | | | | | | PB6 |------| OLED | | | | SCL | | PB7 |-----| SDA | | | | | | PA0 |-----| 按键1 | | | ---------- ----------提示实际接线时注意上拉电阻配置DHT11数据线建议接4.7K上拉I2C总线通常模块已内置上拉电阻2. 开发环境搭建2.1 工具链安装推荐使用STM32CubeIDE一站式解决方案从ST官网下载安装包安装时勾选STM32F1系列支持包安装完成后配置代码格式化风格个人偏好LLVM风格# 验证工具链安装成功的快速方法 $ arm-none-eabi-gcc --version arm-none-eabi-gcc (GNU Arm Embedded Toolchain 10-2020-q4-major)2.2 CubeMX工程配置关键配置步骤时钟树配置HSE 8MHz → PLL → 72MHz系统时钟GPIO设置PB5DHT11数据线开漏输出输入模式PB6/PB7I2C1接口PA0按键输入下拉模式I2C参数标准模式100kHz7位地址模式启用I2C中断// 自动生成的I2C初始化代码片段 hi2c1.Instance I2C1; hi2c1.Init.ClockSpeed 100000; hi2c1.Init.DutyCycle I2C_DUTYCYCLE_2; hi2c1.Init.OwnAddress1 0; hi2c1.Init.AddressingMode I2C_ADDRESSINGMODE_7BIT; hi2c1.Init.DualAddressMode I2C_DUALADDRESS_DISABLE; hi2c1.Init.OwnAddress2 0; hi2c1.Init.GeneralCallMode I2C_GENERALCALL_DISABLE; hi2c1.Init.NoStretchMode I2C_NOSTRETCH_DISABLE;3. DHT11驱动开发3.1 单总线通信协议解析DHT11的典型通信时序阶段主机动作从机响应时间参数启动信号拉低≥18ms等待18-30ms响应信号释放总线拉低80us后拉高80us固定160us数据传输每bit起始拉低50us高电平持续时间区分0/126-28us(0) / 70us(1)// DHT11读取函数实现 uint8_t DHT11_ReadData(DHT11_Data *data) { uint8_t buffer[5] {0}; // 启动信号 DHT11_IO_OUT(); DHT11_LOW(); HAL_Delay(20); DHT11_HIGH(); delay_us(30); // 等待响应 DHT11_IO_IN(); if(DHT11_Read() ! 0) return 1; while(DHT11_Read() 0); while(DHT11_Read() 1); // 接收40bit数据 for(int i0; i5; i) { for(int j0; j8; j) { while(DHT11_Read() 0); // 等待50us低电平结束 delay_us(35); // 关键判别点 if(DHT11_Read() 1) { buffer[i] | (1 (7-j)); while(DHT11_Read() 1); } } } // 校验和验证 if(buffer[0] buffer[1] buffer[2] buffer[3] buffer[4]) { >// 页面切换状态机实现 typedef enum { PAGE_MAIN, PAGE_HISTORY, PAGE_SETTINGS, PAGE_MAX } DisplayPage; void OLED_UpdateDisplay(DisplayPage page) { switch(page) { case PAGE_MAIN: OLED_Clear(); OLED_ShowString(1, 1, Temp: ); OLED_ShowNum(1, 7, data.temperature, 2); OLED_ShowString(2, 1, Humi: ); OLED_ShowNum(2, 7, data.humidity, 2); break; case PAGE_HISTORY: // 历史记录显示实现 break; default: break; } }4.2 图形化显示技巧利用OLED的位图特性实现更丰富的显示效果// 温度计图标定义 const uint8_t temp_icon[] { 0x18, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x24, 0x18 }; void OLED_DrawIcon(uint8_t x, uint8_t y, const uint8_t *icon) { OLED_SetPos(x, y); for(int i0; i32; i) { OLED_WriteData(icon[i]); } }5. 功能扩展实现5.1 按键控制模块通过状态机实现稳定的按键检测typedef struct { GPIO_TypeDef* GPIOx; uint16_t GPIO_Pin; uint8_t prev_state; uint32_t press_time; } Button; void Button_Update(Button *btn) { uint8_t curr_state HAL_GPIO_ReadPin(btn-GPIOx, btn-GPIO_Pin); if(curr_state ! btn-prev_state) { if(curr_state GPIO_PIN_SET) { // 按键释放 if(btn-press_time 20 btn-press_time 1000) { // 短按动作 printf(Button short press\n); } btn-press_time 0; } btn-prev_state curr_state; } else if(curr_state GPIO_PIN_RESET) { // 按键按下 btn-press_time; } } // 使用示例 Button mode_btn {GPIOA, GPIO_PIN_0, GPIO_PIN_SET, 0}; while(1) { Button_Update(mode_btn); HAL_Delay(10); }5.2 数据记录功能利用STM32内部Flash模拟EEPROM存储关键数据#define FLASH_PAGE_SIZE 0x400 // 1KB for F103 #define SETTINGS_ADDR ((uint32_t)0x0800FC00) // 最后一页 void Flash_WriteSettings(void) { FLASH_EraseInitTypeDef erase; uint32_t page_error; HAL_FLASH_Unlock(); erase.TypeErase FLASH_TYPEERASE_PAGES; erase.PageAddress SETTINGS_ADDR; erase.NbPages 1; HAL_FLASHEx_Erase(erase, page_error); uint16_t *data (uint16_t*)settings; for(int i0; isizeof(settings)/2; i) { HAL_FLASH_Program(FLASH_TYPEPROGRAM_HALFWORD, SETTINGS_ADDR i*2, data[i]); } HAL_FLASH_Lock(); }6. 系统优化与调试6.1 低功耗设计通过STM32的睡眠模式降低功耗void Enter_LowPowerMode(void) { // 配置唤醒源如EXTI HAL_PWR_EnableWakeUpPin(PWR_WAKEUP_PIN1); // 进入停止模式 HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI); // 唤醒后重新初始化时钟 SystemClock_Config(); }6.2 常见问题排查DHT11通信失败的可能原因及解决方案无响应检查接线是否正确VCC、GND、DATA确认上拉电阻已连接4.7KΩ测量供电电压3.3V-5V校验和错误增加启动后的延时DHT11需要1s稳定时间检查时序精度特别是微秒级延时避免频繁读取间隔≥1s数据不稳定添加软件滤波如滑动平均检查电源去耦电容建议100nF避免长距离布线建议20cm7. 项目进阶方向7.1 无线传输扩展通过ESP-01S模块增加WiFi功能// AT指令控制示例 void ESP_SendData(float temp, float humi) { UART_SendString(ATCIPSTART\TCP\,\api.thingspeak.com\,80\r\n); Delay_ms(1000); char cmd[128]; sprintf(cmd, GET /update?api_keyXXXfield1%.1ffield2%.1f\r\n, temp, humi); UART_SendString(ATCIPSEND); UART_SendNumber(strlen(cmd)); UART_SendString(\r\n); Delay_ms(500); UART_SendString(cmd); }7.2 外壳设计与电源管理3D打印外壳设计建议预留传感器通风孔考虑OLED可视角度加入挂墙安装孔电源方案选择锂电池充电管理如TP4056太阳能供电系统5V USB电源适配器在完成基础功能后尝试添加更多传感器如CO2、光照或接入智能家居平台将项目升级为真正的环境监测站。