告别串口打印!用STM32+DS18B20做个OLED温湿度计(HAL库+SSD1306)
STM32实战:打造OLED温湿度监测系统(DS18B20+SSD1306)
每次调试嵌入式项目时,盯着串口助手看数据总有种隔靴搔痒的感觉。最近在工作室整理零件时,发现抽屉里还躺着几片0.96寸OLED和DS18B20温度传感器,突然萌生了个想法——何不把这些零散模块组合成可直接观察的桌面温度计?这个看似简单的需求,实际涉及传感器通信协议、显示驱动移植和HAL库高效使用三个技术层面。
1. 硬件架构设计
1.1 元件选型与连接
这次项目核心采用STM32F103C8T6最小系统板(蓝色药丸),搭配DS18B20数字温度传感器和SSD1306驱动的0.96寸OLED显示屏。选择这套组合主要考虑三点:
- 成本控制:整套硬件成本不超过50元
- 开发便捷:都有成熟的HAL库支持
- 功耗优势:整体工作电流<10mA
接线方案如下表所示:
| 模块 | STM32接口 | 备注 |
|---|---|---|
| DS18B20 DQ | PA1 | 需接4.7K上拉电阻 |
| OLED SCL | PB6 | I2C1时钟线 |
| OLED SDA | PB7 | I2C1数据线 |
| OLED VCC | 3.3V | 注意避免5V供电 |
| OLED GND | GND | 共地连接 |
实际焊接时发现,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μs | 450-500μs |
| 存在检测等待 | 60μs | 40-80μs |
| 写0低电平时间 | 60μs | 55-65μs |
| 读采样窗口 | 15μs | 12-18μs |
2.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 i=0; i<FILTER_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, history+1, 127*sizeof(float)); history[127] = temp; for(uint8_t x=0; x<128; 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秒以内。这个小细节让我深刻体会到,嵌入式开发不仅是写代码,还需要考虑物理环境对系统的影响。
