AHT20温湿度传感器在STM32上的应用:从数据采集到OLED显示

news2026/3/28 21:31:51
AHT20温湿度传感器在STM32上的实战应用从数据采集到OLED可视化在物联网和智能硬件开发中环境数据的实时监测与可视化是基础却关键的一环。AHT20作为新一代数字温湿度传感器以其高精度、低功耗和I2C接口的便捷性成为STM32开发者构建环境监测系统的热门选择。本文将完整呈现如何从零开始在STM32平台上实现AHT20的数据采集、处理并通过OLED屏幕进行专业级可视化展示。1. 硬件架构设计与环境搭建1.1 核心组件选型与连接项目所需的核心硬件包括STM32F103C8T6性价比极高的Cortex-M3内核MCUAHT20传感器模块工作电压3.3VI2C地址0x380.96寸OLED显示屏SSD1306驱动128x64分辨率杜邦线用于各模块间的连接硬件连接示意图STM32引脚连接目标备注PB6OLED SCLI2C1时钟线PB7OLED SDAI2C1数据线PB10AHT20 SCL软件模拟I2C时钟线PB11AHT20 SDA软件模拟I2C数据线3.3V模块VCC统一供电GND模块GND共地连接提示AHT20对电源噪声敏感建议在VCC与GND之间添加0.1μF去耦电容1.2 开发环境配置使用STM32CubeIDE作为开发环境需进行以下基础配置创建新工程选择对应STM32型号配置系统时钟为72MHz启用SWD调试接口添加必要的库文件stm32f1xx_hal_i2c.hstm32f1xx_hal_gpio.hstm32f1xx_hal_delay.h// 系统时钟配置示例 void SystemClock_Config(void) { RCC_OscInitTypeDef RCC_OscInitStruct {0}; RCC_ClkInitTypeDef RCC_ClkInitStruct {0}; RCC_OscInitStruct.OscillatorType RCC_OSCILLATORTYPE_HSE; RCC_OscInitStruct.HSEState RCC_HSE_ON; RCC_OscInitStruct.HSEPredivValue RCC_HSE_PREDIV_DIV1; RCC_OscInitStruct.PLL.PLLState RCC_PLL_ON; RCC_OscInitStruct.PLL.PLLSource RCC_PLLSOURCE_HSE; RCC_OscInitStruct.PLL.PLLMUL RCC_PLL_MUL9; HAL_RCC_OscConfig(RCC_OscInitStruct); RCC_ClkInitStruct.ClockType RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK |RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2; RCC_ClkInitStruct.SYSCLKSource RCC_SYSCLKSOURCE_PLLCLK; RCC_ClkInitStruct.AHBCLKDivider RCC_SYSCLK_DIV1; RCC_ClkInitStruct.APB1CLKDivider RCC_HCLK_DIV2; RCC_ClkInitStruct.APB2CLKDivider RCC_HCLK_DIV1; HAL_RCC_ClockConfig(RCC_ClkInitStruct, FLASH_LATENCY_2); }2. AHT20传感器驱动开发2.1 软件模拟I2C实现由于STM32硬件I2C可能存在稳定性问题我们采用GPIO模拟方式// I2C引脚初始化 void AHT20_I2C_Init(void) { GPIO_InitTypeDef GPIO_InitStruct {0}; __HAL_RCC_GPIOB_CLK_ENABLE(); // 配置PB10和PB11为开漏输出 GPIO_InitStruct.Pin GPIO_PIN_10|GPIO_PIN_11; GPIO_InitStruct.Mode GPIO_MODE_OUTPUT_OD; GPIO_InitStruct.Pull GPIO_NOPULL; GPIO_InitStruct.Speed GPIO_SPEED_FREQ_HIGH; HAL_GPIO_Init(GPIOB, GPIO_InitStruct); // 初始置高 HAL_GPIO_WritePin(GPIOB, GPIO_PIN_10|GPIO_PIN_11, GPIO_PIN_SET); } // I2C起始信号 void I2C_Start(void) { SDA_HIGH(); SCL_HIGH(); delay_us(5); SDA_LOW(); delay_us(5); SCL_LOW(); } // 读取AHT20状态字 uint8_t AHT20_Read_Status(void) { uint8_t status 0; I2C_Start(); I2C_Send_Byte(0x71); // 读地址 status I2C_Read_Byte(0); // 不发送ACK I2C_Stop(); return status; }2.2 传感器初始化与校准AHT20首次上电需要执行初始化校准#define AHT20_ADDRESS 0x38 1 void AHT20_Init(void) { uint8_t cmd[3] {0xBE, 0x08, 0x00}; // 发送初始化命令 I2C_Start(); I2C_Send_Byte(AHT20_ADDRESS | 0x00); I2C_Send_Byte(cmd[0]); I2C_Send_Byte(cmd[1]); I2C_Send_Byte(cmd[2]); I2C_Stop(); HAL_Delay(10); // 等待校准完成 // 检查校准状态 uint8_t status AHT20_Read_Status(); if(!(status 0x08)) { // 校准失败处理 Error_Handler(); } }2.3 温湿度数据采集与处理完整的温湿度读取流程void AHT20_Read_Data(float *temperature, float *humidity) { uint8_t data[6] {0}; uint32_t temp_raw 0, humi_raw 0; // 触发测量 uint8_t trigger_cmd[3] {0xAC, 0x33, 0x00}; I2C_Start(); I2C_Send_Byte(AHT20_ADDRESS | 0x00); for(int i0; i3; i) { I2C_Send_Byte(trigger_cmd[i]); } I2C_Stop(); // 等待测量完成 HAL_Delay(80); // 读取数据 I2C_Start(); I2C_Send_Byte(AHT20_ADDRESS | 0x01); for(int i0; i6; i) { data[i] I2C_Read_Byte(i5?0:1); } I2C_Stop(); // 数据处理 humi_raw ((uint32_t)data[1] 12) | ((uint32_t)data[2] 4) | (data[3] 4); temp_raw ((uint32_t)(data[3] 0x0F) 16) | ((uint32_t)data[4] 8) | data[5]; *humidity (float)humi_raw * 100 / 1048576.0; *temperature (float)temp_raw * 200 / 1048576.0 - 50; }注意AHT20每次测量后需要至少2秒的间隔时间连续读取会导致数据不准确3. OLED显示界面设计3.1 SSD1306驱动实现采用硬件I2C驱动OLEDvoid OLED_Init(void) { // 初始化序列 uint8_t init_cmd[] { 0xAE, 0xD5, 0x80, 0xA8, 0x3F, 0xD3, 0x00, 0x40, 0x8D, 0x14, 0x20, 0x00, 0xA1, 0xC8, 0xDA, 0x12, 0x81, 0xCF, 0xD9, 0xF1, 0xDB, 0x40, 0xA4, 0xA6, 0xAF }; HAL_I2C_Mem_Write(hi2c1, 0x78, 0x00, 1, init_cmd, sizeof(init_cmd), 100); OLED_Clear(); } // 显示字符串 void OLED_ShowString(uint8_t x, uint8_t y, char *str) { while(*str) { OLED_ShowChar(x, y, *str); x 8; if(x 120) { x 0; y 2; } str; } } // 显示温湿度数据 void OLED_ShowTempHumidity(float temp, float humi) { char buffer[16]; // 温度显示 sprintf(buffer, Temp:%.1fC, temp); OLED_ShowString(0, 0, buffer); // 湿度显示 sprintf(buffer, Humi:%.1f%%, humi); OLED_ShowString(0, 2, buffer); // 添加简单的图形指示 uint8_t level (uint8_t)(humi / 10); for(uint8_t i0; ilevel; i) { OLED_DrawRectangle(90 i*3, 16, 92 i*3, 48, 1); } }3.2 专业级数据可视化技巧提升显示效果的几种方法动态曲线绘制实现历史数据趋势图#define HISTORY_SIZE 128 float temp_history[HISTORY_SIZE]; uint8_t history_index 0; void Update_Temp_Graph(float temp) { // 更新历史数据 temp_history[history_index] temp; history_index (history_index 1) % HISTORY_SIZE; // 清空图形区域 OLED_Fill(0, 32, 127, 63, 0); // 绘制坐标轴 OLED_DrawLine(0, 48, 127, 48, 1); // 绘制曲线 for(uint8_t i0; iHISTORY_SIZE-1; i) { uint8_t x1 i; uint8_t y1 48 - (uint8_t)(temp_history[i] * 0.8); uint8_t x2 i1; uint8_t y2 48 - (uint8_t)(temp_history[i1] * 0.8); OLED_DrawLine(x1, y1, x2, y2, 1); } }多页面显示通过按键切换不同视图typedef enum { PAGE_MAIN, PAGE_DETAIL, PAGE_HISTORY } DisplayPage; DisplayPage current_page PAGE_MAIN; void Update_Display(float temp, float humi) { switch(current_page) { case PAGE_MAIN: OLED_ShowTempHumidity(temp, humi); break; case PAGE_DETAIL: Show_Detail_Info(temp, humi); break; case PAGE_HISTORY: Update_Temp_Graph(temp); break; } }报警提示功能当数值超出阈值时闪烁显示void Check_Alarm(float temp, float humi) { static uint8_t blink 0; if(temp 30.0 || humi 80.0) { blink !blink; if(blink) { OLED_InvertArea(0, 127, 0, 15); } HAL_Delay(500); } }4. 系统优化与实战技巧4.1 低功耗设计策略传感器间歇工作模式void Enter_LowPower_Mode(void) { // 关闭OLED OLED_Write_Cmd(0xAE); // 设置AHT20为睡眠模式 I2C_Start(); I2C_Send_Byte(AHT20_ADDRESS | 0x00); I2C_Send_Byte(0xB0); I2C_Send_Byte(0x00); I2C_Stop(); // 配置STM32进入STOP模式 HAL_PWR_EnterSTOPMode(PWR_LOWPOWERREGULATOR_ON, PWR_STOPENTRY_WFI); // 唤醒后重新初始化 SystemClock_Config(); OLED_Init(); AHT20_Init(); }动态刷新率调整uint32_t last_update 0; uint32_t update_interval 1000; // 默认1秒 void Adjust_Refresh_Rate(float temp_diff) { // 温度变化大时提高刷新率 if(fabs(temp_diff) 2.0) { update_interval 500; } // 温度稳定时降低刷新率 else if(fabs(temp_diff) 0.5) { update_interval 2000; } else { update_interval 1000; } }4.2 数据滤波与校准移动平均滤波#define FILTER_SIZE 5 float temp_filter[FILTER_SIZE] {0}; uint8_t filter_index 0; float Apply_Filter(float new_value) { temp_filter[filter_index] new_value; filter_index (filter_index 1) % FILTER_SIZE; float sum 0; for(uint8_t i0; iFILTER_SIZE; i) { sum temp_filter[i]; } return sum / FILTER_SIZE; }传感器校准补偿typedef struct { float temp_offset; float humi_offset; } CalibrationParams; CalibrationParams calib {0.5, -2.0}; // 示例校准值 float Apply_Calibration(float temp, float humi) { return temp calib.temp_offset, humi calib.humi_offset; }4.3 异常处理与系统健壮性I2C总线恢复机制void I2C_Recovery(void) { // 发送9个时钟脉冲尝试恢复 for(uint8_t i0; i9; i) { SCL_LOW(); delay_us(5); SCL_HIGH(); delay_us(5); } // 发送STOP条件 I2C_Stop(); // 重新初始化 AHT20_I2C_Init(); AHT20_Init(); }数据有效性检查uint8_t Is_Data_Valid(float temp, float humi) { // 检查数值范围 if(temp -40.0 || temp 85.0) return 0; if(humi 0.0 || humi 100.0) return 0; // 检查变化率是否合理 static float last_temp 25.0; static float last_humi 50.0; if(fabs(temp - last_temp) 5.0) return 0; if(fabs(humi - last_humi) 10.0) return 0; last_temp temp; last_humi humi; return 1; }5. 项目扩展与进阶应用5.1 多传感器数据融合结合其他传感器提升系统功能// 气压传感器BMP280数据读取 float Read_BMP280_Pressure(void) { // 实现气压读取逻辑 return 1013.25; // 示例值 } // 综合环境舒适度计算 float Calculate_Comfort_Index(float temp, float humi) { // 简化的不适指数计算 return temp 0.3 * humi; } void Update_Comfort_Display(void) { float temp, humi; AHT20_Read_Data(temp, humi); float pressure Read_BMP280_Pressure(); float comfort Calculate_Comfort_Index(temp, humi); char buffer[20]; sprintf(buffer, Comfort: %.1f, comfort); OLED_ShowString(0, 4, buffer); // 根据舒适度给出建议 if(comfort 21) { OLED_ShowString(0, 6, Suggestion: Normal); } else if(comfort 25) { OLED_ShowString(0, 6, Suggestion: Warm); } else { OLED_ShowString(0, 6, Suggestion: Hot!); } }5.2 无线数据传输实现通过ESP8266添加Wi-Fi功能void ESP8266_Send_Data(float temp, float humi) { char cmd[128]; sprintf(cmd, ATCIPSTART\TCP\,\api.thingspeak.com\,80\r\n); HAL_UART_Transmit(huart1, (uint8_t*)cmd, strlen(cmd), 1000); sprintf(cmd, GET /update?api_keyYOUR_KEYfield1%.1ffield2%.1f\r\n, temp, humi); uint16_t len strlen(cmd); char send_cmd[64]; sprintf(send_cmd, ATCIPSEND%d\r\n, len); HAL_UART_Transmit(huart1, (uint8_t*)send_cmd, strlen(send_cmd), 1000); HAL_UART_Transmit(huart1, (uint8_t*)cmd, len, 1000); }5.3 本地数据存储方案使用SPI Flash存储历史数据#define FLASH_SECTOR_SIZE 4096 void Save_To_Flash(float temp, float humi) { static uint32_t write_addr 0; // 准备数据 uint8_t data[8]; memcpy(data, temp, 4); memcpy(data4, humi, 4); // 写入Flash HAL_FLASH_Unlock(); FLASH_Erase_Sector(FLASH_SECTOR_5, VOLTAGE_RANGE_3); HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, 0x08020000 write_addr, *(uint32_t*)data); HAL_FLASH_Program(FLASH_TYPEPROGRAM_WORD, 0x08020004 write_addr, *(uint32_t*)(data4)); HAL_FLASH_Lock(); write_addr (write_addr 8) % FLASH_SECTOR_SIZE; }在项目开发过程中调试AHT20时发现其I2C时序要求比常规传感器更为严格特别是在起始条件后的第一个字节传输间隔需要精确控制。通过逻辑分析仪捕获的信号显示在SCL高电平期间保持SDA稳定的时间至少需要4.7μs才能确保可靠通信。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2459252.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…