当前位置: 首页 > news >正文

告别串口打印!用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 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进行初始化配置时,有几个关键点需要特别注意:

  1. 时钟树配置

    HCLK = 72MHz APB1 Prescaler = 2 APB2 Prescaler = 1

    这样配置可确保定时器时钟满足微秒级延时需求。

  2. I2C参数设置

    I2C Mode: I2C Speed: 400kHz Fast Mode Rise Time: 250ns
  3. GPIO模式

    • 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μs

2.2 温度数据处理技巧

DS18B20的原始数据需要特殊处理才能得到实际温度值。在项目中采用了两种显示优化方案:

  1. 滑动平均滤波

    #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; }
  2. 温度单位转换

    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 温度可视化设计

为了提升用户体验,设计了三种显示界面并通过按键切换:

  1. 数字模式

    void show_digital(float temp) { char buf[16]; sprintf(buf, "Temp:%.1fC", temp); OLED_ShowString(0, 2, (uint8_t*)buf, 16); }
  2. 图表模式

    void show_graph(float temp) { uint8_t level = (temp - 15) * 2; // 15-40℃映射到0-50像素 OLED_DrawRectangle(50, 63-level, 30, level); }
  3. 趋势模式

    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. 项目进阶方向

完成基础功能后,可以考虑以下几个扩展方向:

  1. 多传感器网络

    • 通过单总线挂接多个DS18B20
    • 为每个传感器分配ROM地址
    • 轮询显示各节点温度
  2. 无线传输

    // 添加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); }
  3. 历史记录功能

    • 外接AT24Cxx系列EEPROM
    • 每小时记录温度数据
    • 通过按键查看历史极值

实际调试中发现,当环境温度变化剧烈时,传感器响应会有约3秒的延迟。后来在传感器外围加了导热硅胶套,响应速度提升到1秒以内。这个小细节让我深刻体会到,嵌入式开发不仅是写代码,还需要考虑物理环境对系统的影响。

http://www.zskr.cn/news/1327317.html

相关文章:

  • 服务号版本:weixin-java-mp=4.8.3.B,spring-boot=3.3.1,httpclient5=5.5.2
  • 2026全城区上门东营市黄金铂金白银,诚信店铺靠谱TPO1 实力排行榜即地址联系方式 - 亦辰小黄鸭
  • VeriDebug:基于LLM的Verilog智能调试框架解析
  • 如何3秒预览Office文件:QuickLook插件终极指南
  • Python之vyvert包语法、参数和实际应用案例
  • Unity UI布局避坑指南:搞懂LayoutGroup里Control Child Size和Child Force Expand到底怎么选
  • 2026全城区上门鄂州市黄金铂金白银,诚信店铺靠谱TPO1 实力排行榜即地址联系方式 - 亦辰小黄鸭
  • 司法监所:纯视觉无感管控,破解 UWB 标签规避难题
  • Phyphox实验避坑指南:测声速时管长、温度、管口校正那些事儿
  • 从Verilog到GDS:用Calibre nmLVS-H模式搞定复杂芯片的层级化物理验证
  • 2026全城区上门防城港市黄金铂金白银,诚信店铺靠谱TPO1 实力排行榜即地址联系方式 - 亦辰小黄鸭
  • Steam成就管理终极指南:专业工具解锁游戏全成就的深度解析
  • 深耕财税赋能+精准GEO推广 好账本兰宝玺双线发力助企破局
  • 2026全城区上门佛山市黄金铂金白银,诚信店铺靠谱TPO1 实力排行榜即地址联系方式 - 亦辰小黄鸭
  • 在Matlab中绘制横直方图
  • 如何高效使用Alas:碧蓝航线自动化智能助手终极指南
  • 2026德阳市黄金白银铂金回收 全区域正规变现 诚信门店TOP5实力排行榜地址及联系方式_转自TXT - 前途无量YY
  • 2026全城区上门福州市黄金铂金白银,诚信店铺靠谱TPO1 实力排行榜即地址联系方式 - 亦辰小黄鸭
  • 2026安阳市黄金白银铂金回收 全区域正规变现 诚信门店TOP5实力排行榜地址及联系方式_转自TXT - 前途无量YY
  • Codex+Coze自动化工作流实战
  • 量子计算中数据驱动的哈密顿修正方法研究
  • 2026巴中市黄金白银铂金回收 全区域正规变现 诚信门店TOP5实力排行榜地址及联系方式_转自TXT - 前途无量YY
  • 抖音批量下载工具终极指南:3分钟实现无水印高效下载
  • Windows下用VS2019和libusb库,手把手教你写一个控制安卓手机的C++程序(附完整源码)
  • XHS-Downloader:小红书无水印下载终极解决方案,轻松保存优质内容
  • Arm Neoverse N2与CMN-700系统中的PoC与缓存一致性解析
  • AArch64虚拟内存系统架构与转换表描述符详解
  • 从Word到LaTeX:探索docx2tex如何实现学术文档的无缝转换
  • 百度网盘下载加速:用Python脚本突破限速瓶颈的完整指南
  • 3步搭建你的游戏串流魔法:用Sunshine让游戏无处不在