蓝桥杯嵌入式省赛真题解析:STM32G431如何用ADC+定时器实现电压计时器(附完整工程)
STM32G431实战从零构建高精度电压计时器的5个关键步骤在嵌入式系统开发中ADC采集与定时器协同工作是一个经典而实用的技术组合。今天我们就以STM32G431平台为例手把手教你构建一个工业级精度的电压阈值触发计时系统。这个方案不仅适用于蓝桥杯嵌入式竞赛更能直接移植到实际项目中比如电池充放电监控、环境传感器数据采集等场景。1. 硬件架构设计与初始化1.1 核心外设选型与配置STM32G431的ADC模块支持16位分辨率通过过采样实现但在大多数应用中12位精度已经足够。我们需要重点关注以下几个硬件配置点// ADC初始化关键参数 hadc2.Instance ADC2; hadc2.Init.ClockPrescaler ADC_CLOCK_ASYNC_DIV2; hadc2.Init.Resolution ADC_RESOLUTION_12B; hadc2.Init.ScanConvMode ADC_SCAN_DISABLE; hadc2.Init.ContinuousConvMode DISABLE; hadc2.Init.DiscontinuousConvMode DISABLE; hadc2.Init.ExternalTrigConv ADC_SOFTWARE_START; hadc2.Init.DataAlign ADC_DATAALIGN_RIGHT; hadc2.Init.NbrOfConversion 1;定时器配置要点使用基本定时器TIM6产生1Hz中断时钟源选择内部时钟(CK_INT)预分频器根据系统时钟频率计算参数计算示例说明系统时钟170MHzSTM32G431最大频率预分频17000-1得到10kHz计数频率自动重载10000-11秒中断周期1.2 硬件滤波电路设计虽然软件滤波很重要但硬件前端滤波同样不可忽视。推荐在ADC输入前添加RC滤波电路Vin ────╱╲╱╲╱───┬─── ADC输入 1kΩ×3 | 100nF │ GND这种三级RC滤波网络可以有效抑制高频噪声配合软件滤波能达到最佳效果。2. 软件滤波算法实现2.1 移动平均滤波的优化实现原始代码使用了简单的10点平均滤波我们可以改进为加权移动平均滤波#define FILTER_DEPTH 10 typedef struct { float buffer[FILTER_DEPTH]; float weights[FILTER_DEPTH]; // 加权系数 uint8_t index; } ADC_Filter; float weighted_moving_average(ADC_Filter* filter, float new_value) { filter-buffer[filter-index] new_value; filter-index (filter-index 1) % FILTER_DEPTH; float sum 0; float weight_sum 0; for(int i0; iFILTER_DEPTH; i) { sum filter-buffer[i] * filter-weights[i]; weight_sum filter-weights[i]; } return sum / weight_sum; }提示加权系数可以设置为最近的数据权重更大比如使用线性递减权重[10,9,8,...,1]2.2 异常值检测与处理在工业环境中ADC读数可能会受到瞬时干扰我们需要增加异常值检测#define MAX_VOLTAGE_CHANGE 0.5 // 最大合理电压变化率(V/s) float last_valid_voltage 0; float adc_voltage_filter(float raw_voltage) { static uint32_t last_time 0; uint32_t current_time HAL_GetTick(); float time_diff (current_time - last_time) / 1000.0f; if(fabs(raw_voltage - last_valid_voltage) MAX_VOLTAGE_CHANGE * time_diff) { return last_valid_voltage; // 返回上次有效值 } last_valid_voltage raw_voltage; last_time current_time; return raw_voltage; }3. 状态机实现阈值触发逻辑3.1 计时器状态机设计原始代码使用简单的标志位控制我们可以升级为更健壮的状态机typedef enum { STATE_IDLE, STATE_WAIT_FOR_START, STATE_TIMING, STATE_WAIT_FOR_STOP } TimerState; TimerState timer_state STATE_IDLE; void update_timer_state(float voltage) { static float v_min 1.0f; static float v_max 3.0f; switch(timer_state) { case STATE_IDLE: if(voltage v_min) { timer_state STATE_WAIT_FOR_START; } break; case STATE_WAIT_FOR_START: if(voltage v_min) { HAL_TIM_Base_Start_IT(htim6); ucLED | 0x01; Time_Count 0; timer_state STATE_TIMING; } break; case STATE_TIMING: if(voltage v_max) { timer_state STATE_WAIT_FOR_STOP; } break; case STATE_WAIT_FOR_STOP: if(voltage v_max) { HAL_TIM_Base_Stop_IT(htim6); ucLED 0xFE; timer_state STATE_IDLE; } break; } }3.2 防抖处理与滞后设计在阈值检测中引入滞后可以防止临界值附近的抖动#define HYSTERESIS 0.05f // 50mV滞后 if(timer_state STATE_WAIT_FOR_START) { if(voltage (v_min HYSTERESIS)) { // 只有超过最小值滞后量才触发 // 启动计时 } }4. 多任务调度优化4.1 基于时间戳的任务调度原始代码使用简单的延时控制我们可以改进为更精确的调度方式typedef struct { uint32_t interval; uint32_t last_run; void (*task)(void); } Task; Task task_list[] { {100, 0, LED_Proc}, // 每100ms执行 {100, 0, KEY_Proc}, // 每100ms执行 {150, 0, LCD_Proc}, // 每150ms执行 {50, 0, ADC_Proc} // 每50ms执行 }; void scheduler_run(void) { uint32_t now HAL_GetTick(); for(int i0; isizeof(task_list)/sizeof(Task); i) { if(now - task_list[i].last_run task_list[i].interval) { task_list[i].task(); task_list[i].last_run now; } } }4.2 优先级处理机制对于关键任务可以添加优先级机制void scheduler_run(void) { uint32_t now HAL_GetTick(); // 高优先级任务 if(now - adc_task.last_run adc_task.interval) { adc_task.task(); adc_task.last_run now; return; // 本次调度只执行一个高优先级任务 } // 普通优先级任务 for(int i0; iNORMAL_TASK_COUNT; i) { if(now - normal_tasks[i].last_run normal_tasks[i].interval) { normal_tasks[i].task(); normal_tasks[i].last_run now; break; // 每周期只执行一个普通任务 } } }5. 用户界面与参数设置5.1 菜单系统设计原始代码只有两个界面我们可以扩展为完整的菜单系统typedef struct { const char* title; void (*display)(void); void (*handle_key)(uint8_t); } MenuItem; MenuItem menu[] { {Data View, display_data, NULL}, {Set Vmax, display_vmax, adjust_vmax}, {Set Vmin, display_vmin, adjust_vmin}, {Calibrate, display_calib, handle_calib} }; uint8_t current_menu 0; void LCD_Proc(void) { menu[current_menu].display(); } void KEY_Proc(void) { if(menu[current_menu].handle_key) { menu[current_menu].handle_key(key_value); } }5.2 参数存储与加载添加Flash存储功能保存用户设置的参数#define PARAM_ADDR 0x0800F000 // Flash最后一页 typedef struct { float v_max; float v_min; uint32_t crc; } SystemParams; void save_parameters(void) { SystemParams params { .v_max Volt_Max_Active / 10.0f, .v_min Volt_Min_Active / 10.0f }; params.crc calculate_crc(params, sizeof(params)-4); HAL_FLASH_Unlock(); FLASH_Erase_Sector(FLASH_SECTOR_11, VOLTAGE_RANGE_3); uint64_t* p (uint64_t*)params; for(int i0; isizeof(params); i8) { HAL_FLASH_Program(FLASH_TYPEPROGRAM_DOUBLEWORD, PARAM_ADDRi, *p); } HAL_FLASH_Lock(); }注意Flash编程前必须擦除整个扇区且STM32G431的最小编程单位是双字(64位)6. 性能优化与调试技巧6.1 ADC采样时间优化通过调整ADC采样时间可以在速度和精度之间取得平衡// 在ADC通道配置中 sConfig.SamplingTime ADC_SAMPLETIME_47CYCLES_5; // 适用于高阻抗源不同采样时间对精度的影响采样周期适合信号源阻抗转换时间1.510kΩ最短7.550kΩ中等47.550kΩ最长6.2 使用DMA提高效率对于需要高速采样的应用可以启用DMA// 在ADC初始化中添加 hadc2.Init.DMAContinuousRequests ENABLE; hadc2.DMA_Handle hdma_adc2; // 启动带DMA的ADC HAL_ADC_Start_DMA(hadc2, (uint32_t*)adc_buffer, BUFFER_SIZE);6.3 调试输出接口添加SWO调试输出可以方便地监控系统状态void SWO_Print(char* msg) { for(; *msg; msg) { ITM_SendChar(*msg); } } // 在需要调试的地方调用 SWO_Print(Current voltage: ); SWO_Print(float_to_str(ADC_Collected_Data_Aver));在项目实际开发中我发现最常出现问题的环节是阈值检测的逻辑处理。特别是在电压波动较大的环境中单纯的上限下限比较很容易产生误触发。通过引入状态机和滞后处理系统的稳定性得到了显著提升。另一个实用技巧是在ADC输入端添加一个0.1uF的陶瓷电容这能有效抑制高频干扰比单纯依赖软件滤波效果更好。