用STM32F103C8T6打造触摸感应示波器从ADC采集到OLED波形显示的趣味实践在嵌入式开发领域将枯燥的技术参数转化为可视化的交互体验往往能激发学习者的深层兴趣。今天我们要实现的不仅是一个简单的信号采集系统而是一个可以通过手指触摸就能实时观察波形变化的迷你示波器。这个项目完美结合了STM32的ADC功能与OLED的图形显示能力让抽象的电子信号变得触手可及。1. 项目构思与核心组件1.1 为什么选择STM32F103C8T6这款被称为蓝色药丸的开发板以其极高的性价比在创客圈广受欢迎Cortex-M3内核72MHz主频足以处理实时信号12位ADC提供0-3.3V电压范围的精确测量丰富外设多达37个GPIO轻松连接各类传感器和显示器开发生态完善支持Keil、IAR、PlatformIO等多种开发环境1.2 触摸感应的实现原理与传统示波器不同我们的设计采用人体触摸作为信号源// 触摸检测阈值设置 #define TOUCH_THRESHOLD 100 // 根据实际环境调整当手指接触PA7引脚时人体感应的50Hz交流信号会形成微弱的电压变化。ADC通过定期采样捕获这些变化经过软件处理后转化为可视波形。1.3 OLED显示选型建议两种常见规格对比参数0.96寸OLED1.3寸OLED分辨率128×64128×64起始地址0x000x02可视角度160度170度典型功耗0.08W0.1W提示1.3寸屏的初始化代码需要特别注意列地址偏移量这是与0.96寸屏的主要区别。2. 硬件连接与配置2.1 最小系统搭建所需元件清单STM32F103C8T6开发板OLED显示屏SSD1306驱动10kΩ电阻用于PA7下拉杜邦线若干接线示意图PA7 ---[10kΩ]--- GND (触摸感应输入) PA2 --- SCL PA3 --- SDA 3.3V --- VCC GND --- GND2.2 ADC配置关键步骤在CubeMX中的设置要点启用ADC1的Channel7对应PA7设置采样时间为55.5周期配置为连续转换模式启用DMA传输以提高效率对应的初始化代码片段void MX_ADC1_Init(void) { hadc1.Instance ADC1; hadc1.Init.ScanConvMode DISABLE; hadc1.Init.ContinuousConvMode ENABLE; hadc1.Init.DataAlign ADC_DATAALIGN_RIGHT; hadc1.Init.NbrOfConversion 1; if (HAL_ADC_Init(hadc1) ! HAL_OK) { Error_Handler(); } }3. 软件架构设计3.1 主程序流程图开始 ├─ 硬件初始化 │ ├─ ADC配置 │ ├─ OLED初始化 │ └─ 定时器设置 ├─ 进入主循环 │ ├─ ADC采样 │ ├─ 数据处理 │ │ ├─ 均值滤波 │ │ └─ 幅度归一化 │ └─ 波形绘制 └─ (循环执行)3.2 关键算法实现滑动窗口滤波消除触摸信号的随机噪声#define SAMPLE_SIZE 10 uint16_t filter_buffer[SAMPLE_SIZE]; uint16_t adc_filter(uint16_t new_sample) { static uint8_t index 0; filter_buffer[index] new_sample; if(index SAMPLE_SIZE) index 0; uint32_t sum 0; for(int i0; iSAMPLE_SIZE; i){ sum filter_buffer[i]; } return sum / SAMPLE_SIZE; }动态范围调整自动适应不同触摸强度void auto_scale(uint16_t *values, uint8_t count) { uint16_t min 4095, max 0; for(int i0; icount; i){ if(values[i] min) min values[i]; if(values[i] max) max values[i]; } float scale 64.0f / (max - min); for(int i0; icount; i){ values[i] (uint16_t)((values[i] - min) * scale); } }4. 高级功能扩展4.1 触摸灵敏度调节通过长按屏幕特定区域进入设置模式左上角点击降低灵敏度右上角点击提高灵敏度底部点击返回正常模式实现代码框架void check_setting_mode(void) { static uint32_t press_time 0; if(adc_value TOUCH_THRESHOLD){ if(press_time 0) press_time HAL_GetTick(); else if(HAL_GetTick() - press_time 2000){ enter_setting_mode(); press_time 0; } } else { press_time 0; } }4.2 波形捕获与回放添加SD卡模块后可实现按KEY1保存当前波形到SD卡按KEY2回放最近保存的波形按KEY3删除当前记录文件存储格式建议[文件头] 采样率: 1000Hz 数据点: 128 [数据] 235,241,238,245,250,255,260,...4.3 多通道显示优化当使用1.3寸OLED时可以同时显示主窗口实时波形副窗口FFT频谱分析状态栏当前灵敏度/捕获模式屏幕分区示例void draw_ui_layout(void) { // 主波形区 (80x48) OLED_DrawRectangle(0, 0, 127, 47, 1); // FFT频谱区 (80x16) OLED_DrawRectangle(0, 48, 127, 63, 1); // 状态图标 OLED_ShowChar(120, 0, S, 16); }5. 常见问题排查5.1 波形显示不稳定可能原因及解决方案现象排查步骤解决方法波形跳动大检查PA7下拉电阻确保10kΩ电阻可靠连接基线漂移测量开发板GND是否稳定使用示波器检查电源质量触摸无反应确认ADC配置检查CubeMX中的通道设置OLED花屏检查I2C时序调整初始化延迟时间5.2 性能优化技巧降低刷新率将OLED刷新控制在30-50Hz合理使用DMA避免CPU频繁处理ADC数据简化绘图操作只更新变化的像素区域启用编译优化在Keil中设置-O2优化等级注意过度优化可能导致代码可读性下降建议在关键函数添加详细注释。6. 项目进阶方向6.1 添加蓝牙传输功能通过HC-05模块将波形数据发送到手机APP需要配置USART为115200波特率实现简单的数据协议开发Android端接收程序数据传输格式示例# Python解析示例 def parse_data(raw): header raw[0] if header 0xAA: length raw[1] payload raw[2:2length] checksum sum(payload) 0xFF if checksum raw[2length]: return payload return None6.2 引入机器学习识别在STM32上实现简单的波形模式识别收集不同手势的波形样本提取特征值过零率、峰值间隔等实现KNN分类算法特征提取代码片段typedef struct { float zero_crossing; float peak_avg; uint8_t peak_count; } WaveFeature; void extract_feature(uint16_t *data, WaveFeature *out) { uint8_t crosses 0; uint16_t peaks 0; uint8_t count 0; for(int i1; i127; i){ if((data[i-1]-2048)*(data[i]-2048) 0) crosses; if(data[i]data[i-1] data[i]data[i1]){ peaks data[i]; count; } } out-zero_crossing crosses / 128.0f; out-peak_avg count ? (peaks / (float)count) : 0; out-peak_count count; }在完成基础功能后我发现最影响体验的其实是波形刷新率与稳定性的平衡。通过将ADC采样与OLED刷新放在不同的定时器中断中处理并采用双缓冲机制最终实现了既流畅又稳定的显示效果。当看到手指轻触就能在屏幕上产生优美的正弦波时那种即时反馈的成就感正是电子制作的魅力所在。
用STM32F103C8T6做个触摸感应示波器?手把手教你ADC采集+OLED波形显示(附完整代码)
发布时间:2026/5/19 11:17:16
用STM32F103C8T6打造触摸感应示波器从ADC采集到OLED波形显示的趣味实践在嵌入式开发领域将枯燥的技术参数转化为可视化的交互体验往往能激发学习者的深层兴趣。今天我们要实现的不仅是一个简单的信号采集系统而是一个可以通过手指触摸就能实时观察波形变化的迷你示波器。这个项目完美结合了STM32的ADC功能与OLED的图形显示能力让抽象的电子信号变得触手可及。1. 项目构思与核心组件1.1 为什么选择STM32F103C8T6这款被称为蓝色药丸的开发板以其极高的性价比在创客圈广受欢迎Cortex-M3内核72MHz主频足以处理实时信号12位ADC提供0-3.3V电压范围的精确测量丰富外设多达37个GPIO轻松连接各类传感器和显示器开发生态完善支持Keil、IAR、PlatformIO等多种开发环境1.2 触摸感应的实现原理与传统示波器不同我们的设计采用人体触摸作为信号源// 触摸检测阈值设置 #define TOUCH_THRESHOLD 100 // 根据实际环境调整当手指接触PA7引脚时人体感应的50Hz交流信号会形成微弱的电压变化。ADC通过定期采样捕获这些变化经过软件处理后转化为可视波形。1.3 OLED显示选型建议两种常见规格对比参数0.96寸OLED1.3寸OLED分辨率128×64128×64起始地址0x000x02可视角度160度170度典型功耗0.08W0.1W提示1.3寸屏的初始化代码需要特别注意列地址偏移量这是与0.96寸屏的主要区别。2. 硬件连接与配置2.1 最小系统搭建所需元件清单STM32F103C8T6开发板OLED显示屏SSD1306驱动10kΩ电阻用于PA7下拉杜邦线若干接线示意图PA7 ---[10kΩ]--- GND (触摸感应输入) PA2 --- SCL PA3 --- SDA 3.3V --- VCC GND --- GND2.2 ADC配置关键步骤在CubeMX中的设置要点启用ADC1的Channel7对应PA7设置采样时间为55.5周期配置为连续转换模式启用DMA传输以提高效率对应的初始化代码片段void MX_ADC1_Init(void) { hadc1.Instance ADC1; hadc1.Init.ScanConvMode DISABLE; hadc1.Init.ContinuousConvMode ENABLE; hadc1.Init.DataAlign ADC_DATAALIGN_RIGHT; hadc1.Init.NbrOfConversion 1; if (HAL_ADC_Init(hadc1) ! HAL_OK) { Error_Handler(); } }3. 软件架构设计3.1 主程序流程图开始 ├─ 硬件初始化 │ ├─ ADC配置 │ ├─ OLED初始化 │ └─ 定时器设置 ├─ 进入主循环 │ ├─ ADC采样 │ ├─ 数据处理 │ │ ├─ 均值滤波 │ │ └─ 幅度归一化 │ └─ 波形绘制 └─ (循环执行)3.2 关键算法实现滑动窗口滤波消除触摸信号的随机噪声#define SAMPLE_SIZE 10 uint16_t filter_buffer[SAMPLE_SIZE]; uint16_t adc_filter(uint16_t new_sample) { static uint8_t index 0; filter_buffer[index] new_sample; if(index SAMPLE_SIZE) index 0; uint32_t sum 0; for(int i0; iSAMPLE_SIZE; i){ sum filter_buffer[i]; } return sum / SAMPLE_SIZE; }动态范围调整自动适应不同触摸强度void auto_scale(uint16_t *values, uint8_t count) { uint16_t min 4095, max 0; for(int i0; icount; i){ if(values[i] min) min values[i]; if(values[i] max) max values[i]; } float scale 64.0f / (max - min); for(int i0; icount; i){ values[i] (uint16_t)((values[i] - min) * scale); } }4. 高级功能扩展4.1 触摸灵敏度调节通过长按屏幕特定区域进入设置模式左上角点击降低灵敏度右上角点击提高灵敏度底部点击返回正常模式实现代码框架void check_setting_mode(void) { static uint32_t press_time 0; if(adc_value TOUCH_THRESHOLD){ if(press_time 0) press_time HAL_GetTick(); else if(HAL_GetTick() - press_time 2000){ enter_setting_mode(); press_time 0; } } else { press_time 0; } }4.2 波形捕获与回放添加SD卡模块后可实现按KEY1保存当前波形到SD卡按KEY2回放最近保存的波形按KEY3删除当前记录文件存储格式建议[文件头] 采样率: 1000Hz 数据点: 128 [数据] 235,241,238,245,250,255,260,...4.3 多通道显示优化当使用1.3寸OLED时可以同时显示主窗口实时波形副窗口FFT频谱分析状态栏当前灵敏度/捕获模式屏幕分区示例void draw_ui_layout(void) { // 主波形区 (80x48) OLED_DrawRectangle(0, 0, 127, 47, 1); // FFT频谱区 (80x16) OLED_DrawRectangle(0, 48, 127, 63, 1); // 状态图标 OLED_ShowChar(120, 0, S, 16); }5. 常见问题排查5.1 波形显示不稳定可能原因及解决方案现象排查步骤解决方法波形跳动大检查PA7下拉电阻确保10kΩ电阻可靠连接基线漂移测量开发板GND是否稳定使用示波器检查电源质量触摸无反应确认ADC配置检查CubeMX中的通道设置OLED花屏检查I2C时序调整初始化延迟时间5.2 性能优化技巧降低刷新率将OLED刷新控制在30-50Hz合理使用DMA避免CPU频繁处理ADC数据简化绘图操作只更新变化的像素区域启用编译优化在Keil中设置-O2优化等级注意过度优化可能导致代码可读性下降建议在关键函数添加详细注释。6. 项目进阶方向6.1 添加蓝牙传输功能通过HC-05模块将波形数据发送到手机APP需要配置USART为115200波特率实现简单的数据协议开发Android端接收程序数据传输格式示例# Python解析示例 def parse_data(raw): header raw[0] if header 0xAA: length raw[1] payload raw[2:2length] checksum sum(payload) 0xFF if checksum raw[2length]: return payload return None6.2 引入机器学习识别在STM32上实现简单的波形模式识别收集不同手势的波形样本提取特征值过零率、峰值间隔等实现KNN分类算法特征提取代码片段typedef struct { float zero_crossing; float peak_avg; uint8_t peak_count; } WaveFeature; void extract_feature(uint16_t *data, WaveFeature *out) { uint8_t crosses 0; uint16_t peaks 0; uint8_t count 0; for(int i1; i127; i){ if((data[i-1]-2048)*(data[i]-2048) 0) crosses; if(data[i]data[i-1] data[i]data[i1]){ peaks data[i]; count; } } out-zero_crossing crosses / 128.0f; out-peak_avg count ? (peaks / (float)count) : 0; out-peak_count count; }在完成基础功能后我发现最影响体验的其实是波形刷新率与稳定性的平衡。通过将ADC采样与OLED刷新放在不同的定时器中断中处理并采用双缓冲机制最终实现了既流畅又稳定的显示效果。当看到手指轻触就能在屏幕上产生优美的正弦波时那种即时反馈的成就感正是电子制作的魅力所在。