手把手教你用STM32F103C8T6驱动DS18B20,OLED实时显示温度(附完整工程)

张开发
2026/4/15 20:29:25 15 分钟阅读

分享文章

手把手教你用STM32F103C8T6驱动DS18B20,OLED实时显示温度(附完整工程)
STM32F103C8T6与DS18B20温度监测系统实战指南1. 项目概述与硬件准备在嵌入式开发领域温度监测是最基础也最实用的功能之一。使用STM32F103C8T6这款性价比极高的MCU搭配DS18B20数字温度传感器和0.96寸OLED显示屏可以构建一个完整的温度监测系统。这个项目非常适合作为嵌入式开发的入门实践涵盖了GPIO控制、单总线协议、传感器数据采集和显示输出等核心技能。所需硬件清单STM32F103C8T6最小系统板蓝色药丸开发板DS18B20温度传感器防水型或TO-92封装0.96寸OLED显示屏I2C或SPI接口面包板及杜邦线若干4.7kΩ电阻用于DS18B20上拉USB转TTL模块用于程序下载提示DS18B20有寄生电源和工作电源两种模式本教程采用工作电源模式需要连接VCC引脚。硬件连接示意图设备引脚STM32连接引脚DS18B20 VCC3.3VDS18B20 GNDGNDDS18B20 DQPA4OLED VCC3.3VOLED GNDGNDOLED SCLPB6OLED SDAPB72. 开发环境搭建与工程配置2.1 Keil MDK开发环境首先需要安装Keil MDK开发环境并配置STM32F1系列的设备支持包。以下是关键步骤下载并安装Keil MDK-ARM最新版本通过Pack Installer安装STM32F1xx_DFP设备支持包新建工程选择STM32F103C8T6作为目标设备配置工程选项Target选项卡选择正确的晶振频率通常为8MHzOutput选项卡勾选Create HEX FileC/C选项卡添加必要的宏定义如USE_STDPERIPH_DRIVER2.2 硬件抽象层配置我们需要配置三个关键外设GPIO、I2C用于OLED和系统时钟。使用STM32标准外设库可以简化这一过程。// 系统时钟初始化示例 void RCC_Configuration(void) { RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOA | RCC_APB2Periph_GPIOB | RCC_APB2Periph_AFIO, ENABLE); RCC_APB1PeriphClockCmd(RCC_APB1Periph_I2C1, ENABLE); }3. DS18B20驱动实现3.1 单总线协议解析DS18B20采用单总线通信协议这意味着只需要一根数据线加上电源和地即可完成通信。协议的关键在于精确的时序控制。单总线通信基本时序复位脉冲主机拉低总线480μs以上然后释放存在脉冲从机在15-60μs内拉低总线60-240μs写时隙主机拉低总线至少1μs然后在15μs内输出位值读时隙主机拉低总线至少1μs然后在15μs内采样总线状态3.2 底层驱动函数实现以下是DS18B20驱动的核心函数实现// DS18B20初始化函数 uint8_t DS18B20_Init(void) { GPIO_InitTypeDef GPIO_InitStructure; // 配置PA4为推挽输出 GPIO_InitStructure.GPIO_Pin GPIO_Pin_4; GPIO_InitStructure.GPIO_Mode GPIO_Mode_Out_PP; GPIO_InitStructure.GPIO_Speed GPIO_Speed_50MHz; GPIO_Init(GPIOA, GPIO_InitStructure); // 发送复位脉冲 DS18B20_DQ_OUT(0); delay_us(480); // 释放总线并等待存在脉冲 DS18B20_DQ_OUT(1); delay_us(60); // 检测存在脉冲 if(DS18B20_DQ_IN() 0) { delay_us(480); return 1; // 初始化成功 } return 0; // 初始化失败 } // 写入一个字节 void DS18B20_WriteByte(uint8_t dat) { uint8_t i; for(i0; i8; i) { DS18B20_DQ_OUT(0); delay_us(2); if(dat 0x01) DS18B20_DQ_OUT(1); else DS18B20_DQ_OUT(0); delay_us(60); DS18B20_DQ_OUT(1); dat 1; } } // 读取一个字节 uint8_t DS18B20_ReadByte(void) { uint8_t i, dat 0; for(i0; i8; i) { dat 1; DS18B20_DQ_OUT(0); delay_us(2); DS18B20_DQ_OUT(1); delay_us(10); if(DS18B20_DQ_IN()) dat | 0x80; delay_us(50); } return dat; }4. OLED显示驱动实现4.1 OLED初始化0.96寸OLED通常使用SSD1306驱动芯片支持I2C或SPI接口。以下是I2C接口的初始化代码void OLED_Init(void) { OLED_WR_Byte(0xAE, OLED_CMD); // 关闭显示 OLED_WR_Byte(0xD5, OLED_CMD); // 设置显示时钟分频比/振荡器频率 OLED_WR_Byte(0x80, OLED_CMD); OLED_WR_Byte(0xA8, OLED_CMD); // 设置多路复用率 OLED_WR_Byte(0x3F, OLED_CMD); OLED_WR_Byte(0xD3, OLED_CMD); // 设置显示偏移 OLED_WR_Byte(0x00, OLED_CMD); OLED_WR_Byte(0x40, OLED_CMD); // 设置显示开始行 OLED_WR_Byte(0x8D, OLED_CMD); // 电荷泵设置 OLED_WR_Byte(0x14, OLED_CMD); OLED_WR_Byte(0x20, OLED_CMD); // 内存地址模式 OLED_WR_Byte(0x00, OLED_CMD); OLED_WR_Byte(0xA1, OLED_CMD); // 段重映射设置 OLED_WR_Byte(0xC8, OLED_CMD); // 扫描方向设置 OLED_WR_Byte(0xDA, OLED_CMD); // COM引脚硬件配置 OLED_WR_Byte(0x12, OLED_CMD); OLED_WR_Byte(0x81, OLED_CMD); // 对比度设置 OLED_WR_Byte(0xCF, OLED_CMD); OLED_WR_Byte(0xD9, OLED_CMD); // 预充电周期 OLED_WR_Byte(0xF1, OLED_CMD); OLED_WR_Byte(0xDB, OLED_CMD); // VCOMH取消选择级别 OLED_WR_Byte(0x40, OLED_CMD); OLED_WR_Byte(0xA4, OLED_CMD); // 全局显示开启 OLED_WR_Byte(0xA6, OLED_CMD); // 正常显示 OLED_WR_Byte(0xAF, OLED_CMD); // 开启显示 }4.2 温度数据显示在OLED上显示温度数据需要考虑格式化和刷新策略void OLED_ShowTemperature(float temp) { char str[16]; // 温度值格式化 if(temp 0) { sprintf(str, -%02d.%02dC, (int)(-temp), (int)(-temp*100)%100); } else { sprintf(str, %02d.%02dC, (int)temp, (int)(temp*100)%100); } // 显示在OLED上 OLED_ShowString(2, 4, Temperature:); OLED_ShowString(4, 4, str); // 添加温度计图标 OLED_DrawBMP(0, 0, 32, 32, TemperatureIcon); }5. 系统整合与优化5.1 主程序逻辑将各个模块整合到主程序中形成完整的温度监测系统int main(void) { float temperature; // 硬件初始化 SystemInit(); RCC_Configuration(); GPIO_Configuration(); Delay_Init(); OLED_Init(); DS18B20_Init(); // 显示初始界面 OLED_Clear(); OLED_ShowString(1, 1, STM32 Temperature); OLED_ShowString(2, 1, Monitoring System); OLED_ShowString(4, 1, Initializing...); delay_ms(1000); while(1) { // 启动温度转换 DS18B20_Start(); delay_ms(750); // 等待转换完成 // 读取温度值 temperature DS18B20_GetTemp(); // 显示温度 OLED_ClearLine(3); OLED_ClearLine(4); OLED_ShowTemperature(temperature); // 添加简单的温度趋势指示 static float last_temp 0; if(temperature last_temp) { OLED_ShowString(6, 1, Temperature Rising); } else if(temperature last_temp) { OLED_ShowString(6, 1, Temperature Falling); } else { OLED_ShowString(6, 1, Temperature Stable); } last_temp temperature; delay_ms(2000); // 2秒更新一次 } }5.2 常见问题排查在实际开发中可能会遇到以下问题及解决方案问题1DS18B20无响应检查硬件连接是否正确特别是上拉电阻是否接好确保时序控制精确特别是复位和存在脉冲的时序尝试降低通信速度增加延时时间问题2温度读数不稳定确保电源稳定可在VCC和GND之间添加0.1μF电容检查传感器是否接触良好尝试多次读取取平均值问题3OLED显示异常检查I2C地址是否正确通常为0x78或0x7A确保初始化序列完整检查屏幕对比度设置是否合适6. 进阶功能扩展完成基础功能后可以考虑以下扩展方向6.1 温度报警功能// 设置温度报警阈值 #define HIGH_TEMP_ALARM 30.0 #define LOW_TEMP_ALARM 10.0 void CheckTempAlarm(float temp) { if(temp HIGH_TEMP_ALARM) { OLED_ShowString(7, 1, ! HIGH TEMP ALARM !); // 可以添加蜂鸣器报警或LED闪烁 } else if(temp LOW_TEMP_ALARM) { OLED_ShowString(7, 1, ! LOW TEMP ALARM !); } else { OLED_ClearLine(7); } }6.2 温度数据记录添加简单的数据记录功能存储最近24小时的温度数据#define MAX_RECORDS 24 float tempRecords[MAX_RECORDS]; uint8_t recordIndex 0; void RecordTemperature(float temp) { tempRecords[recordIndex] temp; recordIndex (recordIndex 1) % MAX_RECORDS; } void DisplayTempHistory(void) { char str[16]; for(int i0; i8; i) { int idx (recordIndex - 8 i MAX_RECORDS) % MAX_RECORDS; sprintf(str, %2d: %4.1fC, i1, tempRecords[idx]); OLED_ShowString(i1, 10, str); } }6.3 通过串口输出数据添加串口功能将温度数据输出到PCvoid USART_Configuration(void) { USART_InitTypeDef USART_InitStructure; // 配置USART1 USART_InitStructure.USART_BaudRate 115200; USART_InitStructure.USART_WordLength USART_WordLength_8b; USART_InitStructure.USART_StopBits USART_StopBits_1; USART_InitStructure.USART_Parity USART_Parity_No; USART_InitStructure.USART_HardwareFlowControl USART_HardwareFlowControl_None; USART_InitStructure.USART_Mode USART_Mode_Rx | USART_Mode_Tx; USART_Init(USART1, USART_InitStructure); USART_Cmd(USART1, ENABLE); } void SendTempToPC(float temp) { char buffer[32]; sprintf(buffer, Current Temperature: %.2fC\r\n, temp); for(int i0; buffer[i]!0; i) { USART_SendData(USART1, buffer[i]); while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) RESET); } }7. 项目总结与经验分享在实际开发过程中有几个关键点值得特别注意时序精度DS18B20对时序要求严格微秒级延时必须准确。建议使用定时器实现精确延时而非简单的循环延时。电源稳定性DS18B20在温度转换时电流较大电源不稳会导致通信失败。添加去耦电容能显著提高稳定性。显示优化频繁刷新OLED会导致闪烁可以采用局部刷新策略只更新变化的部分。错误处理在实际应用中应该添加完善的错误检测和恢复机制比如通信失败时的重试逻辑。这个项目虽然简单但涵盖了嵌入式开发的多个核心概念包括外设控制、协议实现、数据显示等。通过这个实践开发者可以掌握STM32开发的基本流程为更复杂的项目打下坚实基础。

更多文章