STM32F103C8T6与MLX90614红外测温系统开发实战
1. 项目概述与硬件选型
红外测温技术在工业自动化、医疗设备、智能家居等领域有着广泛应用。本项目基于STM32F103C8T6微控制器和MLX90614红外温度传感器,构建一个高性价比的非接触式温度测量系统。相比传统接触式测温方案,这套系统具有响应速度快、不干扰被测对象、测量范围广等优势。
核心硬件组件:
- 主控芯片:STM32F103C8T6(Cortex-M3内核,72MHz主频,64KB Flash,20KB RAM)
- 温度传感器:MLX90614ESF-BAA(出厂校准,-40℃~125℃物体温度范围,±0.5℃精度)
- 显示模块:0.96寸OLED(SSD1306驱动,128x64分辨率,I2C接口)
- 开发板:蓝色Pill开发板(兼容Arduino接口布局)
硬件连接示意图如下:
| 模块引脚 | STM32连接引脚 | 功能说明 |
|---|---|---|
| MLX90614 SCL | PB6 | I2C1时钟线 |
| MLX90614 SDA | PB7 | I2C1数据线 |
| OLED SCL | PB8 | I2C1时钟线(需分时复用) |
| OLED SDA | PB9 | I2C1数据线(需分时复用) |
注意:实际接线时需确保所有设备的电源极性正确,MLX90614工作电压为3.3V,避免接错导致器件损坏。
2. 开发环境搭建与工程配置
2.1 工具链准备
推荐使用STM32CubeIDE作为开发环境,它集成了STM32CubeMX配置工具和Eclipse IDE,支持一键生成初始化代码。安装步骤如下:
- 从ST官网下载STM32CubeIDE安装包(当前最新版本为1.11.0)
- 运行安装程序,选择默认配置
- 安装完成后,通过Help > STM32CubeIDE Repository安装设备支持包
# 在Linux系统下的快速安装命令 wget https://www.st.com/content/ccc/resource/technical/software/sw_development_suite/group0/0b/05/f0/25/c7/2b/42/9d/stm32cubeide/files/st-stm32cubeide_1.11.0_13639_20220726_1055_amd64.deb_bundle.sh.zip unzip st-stm32cubeide_*.zip chmod +x st-stm32cubeide_*.sh ./st-stm32cubeide_*.sh2.2 工程创建与配置
在STM32CubeIDE中新建工程时,需特别注意以下配置项:
选择正确的芯片型号:STM32F103C8Tx
配置时钟树:
- HSE时钟源:8MHz外部晶振
- 系统时钟:72MHz
- APB1分频:2(36MHz)
- APB2分频:1(72MHz)
外设初始化:
- I2C1:标准模式(100kHz)
- GPIO:PB6/PB7配置为复用开漏输出
- USART1:用于调试输出(可选)
// 典型的I2C初始化代码片段 hi2c1.Instance = I2C1; hi2c1.Init.ClockSpeed = 100000; hi2c1.Init.DutyCycle = I2C_DUTYCYCLE_2; hi2c1.Init.OwnAddress1 = 0; hi2c1.Init.AddressingMode = I2C_ADDRESSINGMODE_7BIT; hi2c1.Init.DualAddressMode = I2C_DUALADDRESS_DISABLE; hi2c1.Init.OwnAddress2 = 0; hi2c1.Init.GeneralCallMode = I2C_GENERALCALL_DISABLE; hi2c1.Init.NoStretchMode = I2C_NOSTRETCH_DISABLE; if (HAL_I2C_Init(&hi2c1) != HAL_OK) { Error_Handler(); }3. MLX90614驱动开发
3.1 SMBus协议实现
MLX90614采用SMBus(System Management Bus)协议,这是I2C协议的子集,主要区别在于:
- 严格的时序要求(时钟速度10kHz-100kHz)
- 必须支持PEC(Packet Error Checking)
- 特定的命令格式和应答机制
关键操作函数:
- 起始条件生成:
void SMBus_Start(void) { HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_SET); // SDA高 HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET); // SCL高 delay_us(5); HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_RESET); // SDA低 delay_us(5); HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); // SCL低 }- 数据读取函数:
uint8_t SMBus_ReadByte(uint8_t ack) { uint8_t byte = 0; for(int i=0; i<8; i++) { HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET); // SCL高 delay_us(2); if(HAL_GPIO_ReadPin(GPIOB, GPIO_PIN_7)) { byte |= (1 << (7-i)); } HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); // SCL低 delay_us(2); } // 发送ACK/NACK HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, ack ? GPIO_PIN_SET : GPIO_PIN_RESET); HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_SET); delay_us(2); HAL_GPIO_WritePin(GPIOB, GPIO_PIN_6, GPIO_PIN_RESET); HAL_GPIO_WritePin(GPIOB, GPIO_PIN_7, GPIO_PIN_SET); // 释放SDA return byte; }3.2 温度数据读取与处理
MLX90614内部RAM中存储的温度数据是经过ADC转换后的原始值,需要按照公式转换为实际温度:
Tobj = (raw_data × 0.02) - 273.15完整的数据读取流程:
- 发送设备地址(0x5A << 1 | 0)
- 发送读取命令(0x07)
- 发送重复起始条件
- 发送设备地址(0x5A << 1 | 1)
- 读取低字节(ACK)
- 读取高字节(ACK)
- 读取PEC(NACK)
- 发送停止条件
float MLX90614_ReadTemp(void) { uint8_t data[2]; uint16_t tempData; SMBus_Start(); SMBus_WriteByte(0x5A << 1); // 设备地址 + 写 SMBus_WriteByte(0x07); // 读取目标温度命令 SMBus_Start(); // 重复起始条件 SMBus_WriteByte((0x5A << 1) | 1); // 设备地址 + 读 data[0] = SMBus_ReadByte(1); // 读取低字节 data[1] = SMBus_ReadByte(1); // 读取高字节 SMBus_ReadByte(0); // 读取PEC SMBus_Stop(); tempData = (data[1] << 8) | data[0]; return (tempData * 0.02) - 273.15; // 转换为摄氏度 }4. OLED显示实现
4.1 SSD1306驱动移植
0.96寸OLED通常使用SSD1306驱动芯片,通过I2C接口通信。由于STM32的I2C外设需要分时复用,建议采用软件模拟I2C的方式实现:
void OLED_WriteCmd(uint8_t cmd) { I2C_Start(); I2C_WriteByte(0x78); // OLED地址 I2C_WriteByte(0x00); // 控制字节 I2C_WriteByte(cmd); // 命令字节 I2C_Stop(); } void OLED_Init(void) { OLED_WriteCmd(0xAE); // 关闭显示 OLED_WriteCmd(0xD5); // 设置时钟分频 OLED_WriteCmd(0x80); OLED_WriteCmd(0xA8); // 设置多路复用率 OLED_WriteCmd(0x3F); // 更多初始化命令... OLED_WriteCmd(0xAF); // 开启显示 }4.2 温度数据显示优化
为提高用户体验,建议在OLED上显示以下信息:
- 当前温度值(大字体)
- 温度单位(℃/℉)
- 测量状态指示
- 历史温度曲线(可选)
void OLED_ShowTemp(float temp) { char str[16]; sprintf(str, "%.1f", temp); OLED_Clear(); OLED_SetFont(&Font16x24); OLED_ShowString(0, 20, str); OLED_SetFont(&Font8x16); OLED_ShowString(90, 24, "C"); OLED_DrawCircle(85, 26, 2); // 度符号 OLED_Refresh(); }5. 系统集成与调试
5.1 主程序逻辑设计
系统应采用状态机架构,确保温度测量和显示不会阻塞系统运行:
typedef enum { STATE_IDLE, STATE_MEASURE, STATE_DISPLAY, STATE_ERROR } SystemState; void main(void) { SystemState state = STATE_IDLE; float temperature = 0; uint32_t lastMeasureTime = 0; HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_I2C1_Init(); OLED_Init(); while(1) { uint32_t now = HAL_GetTick(); switch(state) { case STATE_IDLE: if(now - lastMeasureTime > 1000) { state = STATE_MEASURE; } break; case STATE_MEASURE: temperature = MLX90614_ReadTemp(); lastMeasureTime = now; state = STATE_DISPLAY; break; case STATE_DISPLAY: OLED_ShowTemp(temperature); state = STATE_IDLE; break; case STATE_ERROR: OLED_ShowError(); break; } } }5.2 常见问题排查
问题1:I2C通信失败
- 检查硬件连接是否正确
- 用逻辑分析仪抓取I2C波形
- 确认上拉电阻值(通常4.7kΩ)
问题2:温度读数不稳定
- 确保传感器与被测物体距离适当(2-5cm最佳)
- 检查电源是否稳定(建议增加10μF电容)
- 避免强光直射传感器
问题3:OLED显示异常
- 确认I2C地址是否正确(通常0x78或0x7A)
- 检查初始化序列是否完整
- 确保供电电压在3.3V±10%范围内
6. 进阶功能扩展
6.1 温度报警功能
通过STM32的GPIO驱动蜂鸣器或LED,实现超温报警:
#define TEMP_THRESHOLD 38.0 // 报警阈值 void CheckTempAlert(float temp) { static uint8_t alertState = 0; if(temp > TEMP_THRESHOLD) { if(!alertState) { HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_SET); // 开启蜂鸣器 alertState = 1; } } else { if(alertState) { HAL_GPIO_WritePin(GPIOA, GPIO_PIN_0, GPIO_PIN_RESET); // 关闭蜂鸣器 alertState = 0; } } }6.2 数据记录与导出
利用STM32内部Flash或外接SPI Flash存储历史数据:
#define MAX_RECORDS 100 typedef struct { float temp; uint32_t timestamp; } TempRecord; TempRecord records[MAX_RECORDS]; uint8_t recordIndex = 0; void SaveTempRecord(float temp) { if(recordIndex >= MAX_RECORDS) { recordIndex = 0; } records[recordIndex].temp = temp; records[recordIndex].timestamp = HAL_GetTick(); recordIndex++; }6.3 无线传输功能
通过HC-05蓝牙模块或ESP8266 WiFi模块实现温度数据远程监控:
void SendTempViaUART(float temp) { char buffer[32]; sprintf(buffer, "TEMP:%.1fC\r\n", temp); HAL_UART_Transmit(&huart1, (uint8_t*)buffer, strlen(buffer), 100); }