STM32F103C8T6连接ZH03B传感器,手把手教你做一个桌面PM2.5监测仪(附完整代码)

张开发
2026/4/17 1:05:18 15 分钟阅读

分享文章

STM32F103C8T6连接ZH03B传感器,手把手教你做一个桌面PM2.5监测仪(附完整代码)
STM32F103C8T6与ZH03B传感器实战打造高精度桌面PM2.5监测系统最近工作室的空气质量总让我隐隐担忧尤其是看到窗外雾蒙蒙的天空时。作为硬件爱好者我决定用STM32F103C8T6和ZH03B激光粉尘传感器搭建一个实时监测装置。这个不到巴掌大的小盒子现在已经成为我办公桌上最实用的健康助手每当PM2.5数值超过安全阈值它就会用醒目的红色LED提醒我该开空气净化器了。1. 硬件选型与设计思路选择STM32F103C8T6作为主控是因为它完美平衡了性能与成本。这款Cortex-M3内核的MCU运行在72MHz主频下内置64KB Flash和20KB SRAM完全能够胜任传感器数据处理任务。更重要的是它具备多个USART接口可以同时连接传感器和调试终端。ZH03B是炜盛科技推出的激光散射式PM2.5传感器相比传统红外传感器有三大优势精度高最小检测粒径0.3μm误差范围±10%响应快数据更新周期仅1秒数字输出UART接口直接输出浓度值免去复杂的模拟信号处理硬件配置清单组件型号数量备注主控板STM32F103C8T61蓝色pill开发板传感器ZH03B1需5V供电显示屏SSD1306 OLED1I2C接口128x64分辨率连接线Dupont线若干建议使用彩色线区分功能提示ZH03B工作时电流约100mA建议使用独立电源模块供电避免开发板USB供电不足导致数据异常。2. 硬件连接与电路设计实际接线时发现几个容易踩坑的细节。首先是电源部分最初我直接用开发板的5V输出给传感器供电结果串口数据经常出现乱码。后来用万用表测量发现当传感器启动时电压会瞬间跌落至4.3V。解决方法是在传感器VCC与GND之间并联一个100μF的电解电容。核心连接方案电源部分开发板5V → 100μF电容 → ZH03B的VCC共地连接必须可靠数据通信ZH03B_TXD → STM32的PA3(USART2_RX)注意不连接ZH03B_RXD因为我们只需要接收数据显示模块OLED_SCL → PB6(I2C1_SCL)OLED_SDA → PB7(I2C1_SDA)// 引脚功能映射 #define ZH03B_UART USART2 #define ZH03B_RX_PIN GPIO_Pin_3 #define ZH03B_GPIO_PORT GPIOA #define OLED_I2C I2C1电路搭建完成后建议先用逻辑分析仪检查传感器输出。正常工作时ZH03B会每秒发送一帧包含PM1.0、PM2.5、PM10浓度的数据包波特率默认为9600bps。3. 传感器数据解析实战ZH03B的数据协议是项目中最需要仔细处理的部分。通过示波器抓取到的原始数据帧如下格式0x42 0x4D 0x00 0x1C [数据区] 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 0x00 Checksum关键数据位于第6-13字节字节6-7PM1.0浓度μg/m³字节8-9PM2.5浓度μg/m³字节10-11PM10浓度μg/m³校验和计算方法前30字节的累加和不包括自身// 数据帧结构体定义 typedef struct { uint8_t header[2]; // 0x42 0x4D uint16_t length; uint16_t pm1_0; uint16_t pm2_5; uint16_t pm10; // ...其他字段省略 } ZH03B_Frame;在USART2中断服务程序中我采用状态机方式解析数据void USART2_IRQHandler(void) { static uint8_t rx_buffer[32]; static uint8_t state 0; static uint8_t index 0; if(USART_GetITStatus(USART2, USART_IT_RXNE) ! RESET) { uint8_t byte USART_ReceiveData(USART2); switch(state) { case 0: // 等待帧头0x42 if(byte 0x42) { rx_buffer[index] byte; state 1; } break; case 1: // 等待帧头0x4D if(byte 0x4D) { rx_buffer[index] byte; state 2; } else { state 0; index 0; } break; case 2: // 接收数据区 rx_buffer[index] byte; if(index 32) { process_pm_data(rx_buffer); state 0; index 0; } break; } } }注意传感器在刚上电时需要约30秒的预热时间此时输出的数据可能不稳定建议在软件中加入启动延时。4. OLED显示界面优化为了让数据展示更直观我为SSD1306显示屏设计了三级可视化方案数字模式直接显示实时数值PM2.5: 56 μg/m³ PM10: 89 μg/m³ Updated: 12:30:45图表模式绘制最近10分钟的趋势图预警模式当PM2.5超过75时切换红色背景使用u8g2库驱动OLED的示例代码#include u8g2.h u8g2_t u8g2; void oled_init(void) { u8g2_Setup_ssd1306_i2c_128x64_noname_f(u8g2, U8G2_R0, u8x8_byte_sw_i2c, u8x8_gpio_and_delay_stm32); u8g2_InitDisplay(u8g2); u8g2_SetPowerSave(u8g2, 0); } void show_pm_value(uint16_t pm25) { char buf[20]; u8g2_ClearBuffer(u8g2); // 设置预警颜色 if(pm25 75) { u8g2_SetDrawColor(u8g2, 1); u8g2_DrawBox(u8g2, 0, 0, 128, 64); u8g2_SetDrawColor(u8g2, 0); } else { u8g2_SetDrawColor(u8g2, 1); } u8g2_SetFont(u8g2, u8g2_font_profont29_tf); sprintf(buf, %d, pm25); u8g2_DrawStr(u8g2, 40, 35, buf); u8g2_SetFont(u8g2, u8g2_font_profont12_tf); u8g2_DrawStr(u8g2, 50, 50, μg/m³); u8g2_SendBuffer(u8g2); }实际测试中发现频繁刷新全屏会导致显示闪烁。优化方案是采用局部刷新策略只更新数值变化的部分区域。5. 系统集成与性能优化将各模块整合后需要考虑三个关键问题实时性、功耗和稳定性。我的解决方案是实时性保障设置USART2中断优先级为最高使用DMA传输显示数据采用RTOS创建独立任务处理传感器数据// FreeRTOS任务配置 void vSensorTask(void *pvParameters) { while(1) { if(xQueueReceive(pm25_queue, current_pm25, portMAX_DELAY)) { update_display(current_pm25); check_alert_threshold(current_pm25); } } } void vUARTTask(void *pvParameters) { // 串口数据处理逻辑 }功耗控制空闲时让MCU进入STOP模式通过传感器EN引脚控制供电非连续监测场景降低OLED刷新率至2Hz稳定性增强措施数据校验除了校验和还增加帧长度和魔数验证异常处理连续3次校验失败后自动复位传感器滑动滤波对PM2.5数据做5点中值滤波#define FILTER_WINDOW 5 uint16_t median_filter(uint16_t new_value) { static uint16_t buffer[FILTER_WINDOW] {0}; static uint8_t index 0; buffer[index] new_value; if(index FILTER_WINDOW) index 0; // 排序找中值 uint16_t temp[FILTER_WINDOW]; memcpy(temp, buffer, sizeof(temp)); bubble_sort(temp, FILTER_WINDOW); return temp[FILTER_WINDOW/2]; }6. 扩展功能实现基础功能稳定后可以进一步增加实用特性空气质量评估算法typedef enum { AIR_EXCELLENT, // 0-35 AIR_GOOD, // 36-75 AIR_MODERATE, // 76-115 AIR_UNHEALTHY // 115 } AirQualityLevel; AirQualityLevel evaluate_air(uint16_t pm25) { if(pm25 35) return AIR_EXCELLENT; else if(pm25 75) return AIR_GOOD; else if(pm25 115) return AIR_MODERATE; else return AIR_UNHEALTHY; }数据记录功能 利用STM32内部Flash模拟EEPROM每小时存储一次数据#define LOG_ADDR 0x0801F000 // Flash最后一页起始地址 void save_to_flash(uint16_t pm25) { static uint32_t offset 0; uint16_t data pm25 | (RTC_GetCounter() 16); FLASH_Unlock(); FLASH_ProgramHalfWord(LOG_ADDR offset, data); FLASH_Lock(); offset 2; if(offset 1024) offset 0; // 循环写入 }无线传输模块可选添加ESP8266实现WiFi上传使用HC-05模块蓝牙传输通过NRF24L01建立私有无线网络// ESP8266数据传输示例 void wifi_send_data(uint16_t pm25) { char cmd[64]; sprintf(cmd, ATCIPSTART\TCP\,\api.thingspeak.com\,80); send_at_command(cmd); sprintf(cmd, GET /update?api_keyXXXfield1%d, pm25); send_at_command(cmd); }这个项目最让我满意的是它的可扩展性——上周我又增加了温湿度传感器和甲醛检测模块现在它已经进化成完整的室内环境监测站。整个开发过程中最耗时的不是编码而是传感器数据的可靠性处理前后调整了5次滤波算法才得到稳定的读数。

更多文章