告别串口打印!用STM32+DS18B20做个OLED温湿度计(HAL库+SSD1306)
STM32实战打造OLED温湿度监测系统DS18B20SSD1306每次调试嵌入式项目时盯着串口助手看数据总有种隔靴搔痒的感觉。最近在工作室整理零件时发现抽屉里还躺着几片0.96寸OLED和DS18B20温度传感器突然萌生了个想法——何不把这些零散模块组合成可直接观察的桌面温度计这个看似简单的需求实际涉及传感器通信协议、显示驱动移植和HAL库高效使用三个技术层面。1. 硬件架构设计1.1 元件选型与连接这次项目核心采用STM32F103C8T6最小系统板蓝色药丸搭配DS18B20数字温度传感器和SSD1306驱动的0.96寸OLED显示屏。选择这套组合主要考虑三点成本控制整套硬件成本不超过50元开发便捷都有成熟的HAL库支持功耗优势整体工作电流10mA接线方案如下表所示模块STM32接口备注DS18B20 DQPA1需接4.7K上拉电阻OLED SCLPB6I2C1时钟线OLED SDAPB7I2C1数据线OLED VCC3.3V注意避免5V供电OLED GNDGND共地连接实际焊接时发现DS18B20的供电电压范围较宽3V-5.5V但OLED屏的SSD1306控制器对电压敏感建议统一使用3.3V供电以避免意外损坏。1.2 CubeMX基础配置使用STM32CubeMX进行初始化配置时有几个关键点需要特别注意时钟树配置HCLK 72MHz APB1 Prescaler 2 APB2 Prescaler 1这样配置可确保定时器时钟满足微秒级延时需求。I2C参数设置I2C Mode: I2C Speed: 400kHz Fast Mode Rise Time: 250nsGPIO模式DS18B20接口配置为开漏输出模式启用I2C引脚的重映射功能如有需要2. 传感器驱动开发2.1 DS18B20时序优化原始的单总线协议实现常因延时不准导致读取失败。经过多次实测总结出更稳定的时序控制方案// 微秒级延时优化版 void delay_us(uint32_t us) { uint32_t start DWT-CYCCNT; uint32_t cycles us * (SystemCoreClock / 1000000); while((DWT-CYCCNT - start) cycles); }关键时序参数调整如下操作标准时长实测稳定范围复位脉冲480μs450-500μs存在检测等待60μs40-80μs写0低电平时间60μs55-65μs读采样窗口15μs12-18μs2.2 温度数据处理技巧DS18B20的原始数据需要特殊处理才能得到实际温度值。在项目中采用了两种显示优化方案滑动平均滤波#define FILTER_LEN 5 float temp_history[FILTER_LEN]; float get_filtered_temp() { static uint8_t index 0; temp_history[index] DS18B20_Get_Temp(); index (index 1) % FILTER_LEN; float sum 0; for(uint8_t i0; iFILTER_LEN; i) { sum temp_history[i]; } return sum / FILTER_LEN; }温度单位转换float convert_to_fahrenheit(float celsius) { return celsius * 1.8 32; }3. OLED显示实现3.1 SSD1306驱动移植市面上常见的SSD1306驱动库往往过于臃肿。经过精简优化核心显示函数仅需以下几个void OLED_Init(void) { uint8_t init_cmds[] { 0xAE, 0xD5, 0x80, 0xA8, 0x3F, 0xD3, 0x00, 0x40, 0x8D, 0x14, 0x20, 0x00, 0xA1, 0xC8, 0xDA, 0x12, 0x81, 0xCF, 0xD9, 0xF1, 0xDB, 0x30, 0xA4, 0xA6, 0xAF }; HAL_I2C_Mem_Write(hi2c1, 0x78, 0x00, 1, init_cmds, sizeof(init_cmds), 100); } void OLED_DisplayOn(void) { uint8_t cmd 0xAF; HAL_I2C_Mem_Write(hi2c1, 0x78, 0x00, 1, cmd, 1, 100); }3.2 温度可视化设计为了提升用户体验设计了三种显示界面并通过按键切换数字模式void show_digital(float temp) { char buf[16]; sprintf(buf, Temp:%.1fC, temp); OLED_ShowString(0, 2, (uint8_t*)buf, 16); }图表模式void show_graph(float temp) { uint8_t level (temp - 15) * 2; // 15-40℃映射到0-50像素 OLED_DrawRectangle(50, 63-level, 30, level); }趋势模式float history[128]; void update_trend(float temp) { memmove(history, history1, 127*sizeof(float)); history[127] temp; for(uint8_t x0; x128; x) { uint8_t y 63 - (history[x] - 15)*2; OLED_DrawPoint(x, y); } }4. 系统集成与优化4.1 低功耗设计通过以下措施将系统功耗从12mA降至3.8mA将OLED刷新率从60Hz降至10Hz启用DS18B20的寄生供电模式调整MCU运行频率至48MHz在温度稳定时进入STOP模式void enter_low_power_mode(void) { HAL_I2C_DeInit(hi2c1); HAL_SuspendTick(); HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI); SystemClock_Config(); HAL_ResumeTick(); MX_I2C1_Init(); }4.2 报警功能扩展增加两个按键设置温度阈值当温度超限时OLED显示闪烁并触发蜂鸣器#define ALARM_HIGH 30.0 #define ALARM_LOW 10.0 void check_alarm(float temp) { static uint8_t blink; if(temp ALARM_HIGH || temp ALARM_LOW) { blink ^ 1; if(blink) OLED_DisplayOff(); else OLED_DisplayOn(); BEEP_ON(); } else { OLED_DisplayOn(); BEEP_OFF(); } }5. 项目进阶方向完成基础功能后可以考虑以下几个扩展方向多传感器网络通过单总线挂接多个DS18B20为每个传感器分配ROM地址轮询显示各节点温度无线传输// 添加HC-12模块代码 void send_wireless_data(float temp) { char msg[32]; sprintf(msg, TEMP:%.1f, temp); HAL_UART_Transmit(huart1, (uint8_t*)msg, strlen(msg), 100); }历史记录功能外接AT24Cxx系列EEPROM每小时记录温度数据通过按键查看历史极值实际调试中发现当环境温度变化剧烈时传感器响应会有约3秒的延迟。后来在传感器外围加了导热硅胶套响应速度提升到1秒以内。这个小细节让我深刻体会到嵌入式开发不仅是写代码还需要考虑物理环境对系统的影响。