用STM32和PID算法做个数控电源:从BUCK电路到双闭环控制的完整实战
用STM32和PID算法打造高精度数控电源从硬件设计到双闭环控制的实战指南在电子制作和嵌入式开发领域一个稳定可靠的电源系统往往是项目成功的基础。对于电子爱好者和嵌入式开发者来说自己动手打造一台数控电源不仅能满足个性化需求更是一次难得的嵌入式系统与电力电子技术融合的实践机会。本文将带你从零开始使用STM32微控制器和PID控制算法构建一个具备电压电流双闭环控制的高性能BUCK型数控电源。1. 项目整体规划与硬件选型任何电子设计项目的第一步都是明确需求和规划系统架构。对于我们的数控电源项目核心目标是实现以下功能输入电压范围12-24V DC输出电压范围0-15V可调输出电流能力0-3A可调电压调节精度±10mV电流调节精度±20mA过压、过流保护功能关键硬件组件选型指南组件类别推荐型号关键参数选型考虑主控MCUSTM32F103C8T672MHz Cortex-M3, 64KB Flash性价比高PWM和ADC资源丰富功率MOSFETIRF540N100V, 33A, Rds(on)44mΩ导通电阻低开关特性好续流二极管MBR20100CT20A, 100V Schottky低正向压降减少损耗储能电感自制或成品100μH, 5A饱和电流确保CCM模式工作电流检测INA199A2双向, 增益50V/V高精度电流检测提示电感选择是BUCK电路设计的关键建议使用铁硅铝磁环手工绕制便于调整电感量。实际制作前最好用LCR表测量确认电感值。硬件设计中最容易忽视的是PCB布局功率回路最小化缩短MOSFET、电感和二极管之间的连线地平面分割将功率地和信号地分开单点连接退耦电容在MCU和运放电源引脚就近放置100nF陶瓷电容散热设计为MOSFET和二极管预留足够的铜箔面积2. BUCK电路原理与工作模式分析BUCK降压电路作为数控电源的核心功率转换部分其工作原理直接影响整个系统的效率和稳定性。不同于简单的线性稳压开关电源通过高频开关和电感储能来实现电压转换效率通常能达到85%以上。BUCK电路三种工作模式对比// 工作模式判断伪代码 if (电感电流始终 0) { 模式 CCM; // 连续导通模式 } else if (电感电流刚好降到零) { 模式 BCM; // 边界导通模式 } else { 模式 DCM; // 不连续导通模式 }在实际设计中我们通常让电路工作在CCM模式因为输出电压纹波更小电流应力更低更容易实现稳定的闭环控制计算临界电感值的公式$$ L_{critical} \frac{(V_{in} - V_{out}) \times V_{out}}{2 \times f_{sw} \times V_{in} \times I_{out_min}} $$其中$f_{sw}$ 是开关频率我们使用50kHz$I_{out_min}$ 是最小输出电流设为100mA代入我们的设计参数$V_{in}$24V$V_{out}$12V$f_{sw}$50kHz$I_{out_min}$0.1A计算得到$L_{critical}$≈72μH因此选择100μH电感可确保在大部分工作条件下处于CCM模式。3. STM32外围电路设计与配置STM32作为控制核心需要正确配置多个外设以实现完整的电源控制功能。以下是关键外设的配置要点3.1 PWM生成配置BUCK电路需要精确的PWM信号控制MOSFET开关。STM32的定时器非常适合此任务// PWM初始化代码示例 void PWM_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseStructure; TIM_OCInitTypeDef TIM_OCInitStructure; RCC_APB1PeriphClockCmd(RCC_APB1Periph_TIM3, ENABLE); // 定时器基础配置 TIM_TimeBaseStructure.TIM_Period 999; // 50kHz PWM 50MHz时钟 TIM_TimeBaseStructure.TIM_Prescaler 0; TIM_TimeBaseStructure.TIM_ClockDivision 0; TIM_TimeBaseStructure.TIM_CounterMode TIM_CounterMode_Up; TIM_TimeBaseInit(TIM3, TIM_TimeBaseStructure); // PWM通道配置 TIM_OCInitStructure.TIM_OCMode TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OutputState TIM_OutputState_Enable; TIM_OCInitStructure.TIM_Pulse 0; // 初始占空比0% TIM_OCInitStructure.TIM_OCPolarity TIM_OCPolarity_High; TIM_OC2Init(TIM3, TIM_OCInitStructure); TIM_Cmd(TIM3, ENABLE); TIM_CtrlPWMOutputs(TIM3, ENABLE); }3.2 ADC采样配置电压电流的精确采样是闭环控制的基础。需要注意使用定时器触发ADC实现固定频率采样配置DMA传输采样结果减少CPU开销合理设置采样保持时间确保精度// ADC多通道DMA配置示例 void ADC_Config(void) { ADC_InitTypeDef ADC_InitStructure; DMA_InitTypeDef DMA_InitStructure; // DMA配置 DMA_DeInit(DMA1_Channel1); DMA_InitStructure.DMA_PeripheralBaseAddr (uint32_t)ADC1-DR; DMA_InitStructure.DMA_MemoryBaseAddr (uint32_t)ADC_Values; DMA_InitStructure.DMA_DIR DMA_DIR_PeripheralSRC; DMA_InitStructure.DMA_BufferSize 3; // 3个通道 DMA_InitStructure.DMA_PeripheralInc DMA_PeripheralInc_Disable; DMA_InitStructure.DMA_MemoryInc DMA_MemoryInc_Enable; DMA_InitStructure.DMA_PeripheralDataSize DMA_PeripheralDataSize_HalfWord; DMA_InitStructure.DMA_MemoryDataSize DMA_MemoryDataSize_HalfWord; DMA_InitStructure.DMA_Mode DMA_Mode_Circular; DMA_InitStructure.DMA_Priority DMA_Priority_High; DMA_InitStructure.DMA_M2M DMA_M2M_Disable; DMA_Init(DMA1_Channel1, DMA_InitStructure); DMA_Cmd(DMA1_Channel1, ENABLE); // ADC配置 ADC_InitStructure.ADC_Mode ADC_Mode_Independent; ADC_InitStructure.ADC_ScanConvMode ENABLE; ADC_InitStructure.ADC_ContinuousConvMode ENABLE; ADC_InitStructure.ADC_ExternalTrigConv ADC_ExternalTrigConv_T3_TRGO; ADC_InitStructure.ADC_DataAlign ADC_DataAlign_Right; ADC_InitStructure.ADC_NbrOfChannel 3; ADC_Init(ADC1, ADC_InitStructure); // 配置采样通道和顺序 ADC_RegularChannelConfig(ADC1, ADC_Channel_0, 1, ADC_SampleTime_55Cycles5); // 输出电压 ADC_RegularChannelConfig(ADC1, ADC_Channel_1, 2, ADC_SampleTime_55Cycles5); // 输出电流 ADC_RegularChannelConfig(ADC1, ADC_Channel_2, 3, ADC_SampleTime_55Cycles5); // 输入电压 ADC_DMACmd(ADC1, ENABLE); ADC_Cmd(ADC1, ENABLE); // 校准ADC ADC_ResetCalibration(ADC1); while(ADC_GetResetCalibrationStatus(ADC1)); ADC_StartCalibration(ADC1); while(ADC_GetCalibrationStatus(ADC1)); ADC_ExternalTrigConvCmd(ADC1, ENABLE); ADC_SoftwareStartConvCmd(ADC1, ENABLE); }4. 双闭环PID控制算法实现数控电源的核心在于闭环控制。我们采用电压外环、电流内环的双闭环结构既能保证输出电压稳定又能提供精确的电流限制和保护。4.1 PID算法代码实现增量式PID算法适合嵌入式系统实现计算量小且不会产生积分饱和问题typedef struct { float kp, ki, kd; // PID参数 float dt; // 采样周期(秒) float error; // 当前误差 float prevError1; // 前一次误差 float prevError2; // 前两次误差 float outLast1; // 上一次输出 float inc; // 增量输出 float iLimitHigh; // 积分上限 float iLimitLow; // 积分下限 } PidObject; void pidInit(PidObject* pid, float kp, float ki, float kd, float dt) { pid-kp kp; pid-ki ki; pid-kd kd; pid-dt dt; // 预计算系数减少实时计算量 pid-ap pid-kp * (1 pid-dt / pid-ki pid-kd / pid-dt); pid-ai pid-kp * (1 2 * pid-kd / pid-dt); pid-ad pid-kp * pid-kd / pid-dt; pidClear(pid); } float pidUpdate(PidObject* pid, const float desired, const float measured) { pid-error desired - measured; // 增量计算 pid-inc (pid-ap * pid-error) - (pid-ai * pid-prevError1) (pid-ad * pid-prevError2); // 输出计算 float output pid-inc pid-outLast1; // 输出限幅 if(output pid-iLimitHigh) { output pid-iLimitHigh; } else if(output pid-iLimitLow) { output pid-iLimitLow; } // 更新状态 pid-prevError2 pid-prevError1; pid-prevError1 pid-error; pid-outLast1 output; return output; }4.2 控制状态机实现数控电源需要根据工作状态切换控制模式典型的状态包括恒压模式(CV)优先稳定输出电压恒流模式(CC)当电流达到设定值时切换保护模式检测到异常时关闭输出typedef enum { CV_MODE, // 恒压模式 CC_MODE, // 恒流模式 FAULT_MODE // 故障模式 } ControlMode; void ControlTask(void) { static uint32_t tick 0; float vo_measured ADC_GetVoltage(); float io_measured ADC_GetCurrent(); // 保护检测 if(vo_measured VO_MAX || io_measured IO_MAX) { PWM_Disable(); currentMode FAULT_MODE; return; } // 正常控制 switch(currentMode) { case CV_MODE: pwmDuty pidUpdate(vPID, vo_setpoint, vo_measured); if(io_measured io_setpoint) { currentMode CC_MODE; pidReset(vPID); } break; case CC_MODE: pwmDuty pidUpdate(iPID, io_setpoint, io_measured); if(vo_measured vo_setpoint - 0.1f) { currentMode CV_MODE; pidReset(iPID); } break; case FAULT_MODE: // 故障恢复逻辑 if(tick RECOVERY_TICKS) { pidClear(vPID); pidClear(iPID); PWM_Enable(); currentMode CV_MODE; tick 0; } break; } PWM_SetDuty(pwmDuty); }5. 系统调试与参数整定完成硬件制作和软件编写后系统调试是确保电源性能的关键步骤。调试应分阶段进行5.1 开环测试流程PWM测试不接功率电路用示波器确认PWM波形正常驱动测试接MOSFET但不接电感检查栅极驱动波形轻载测试接完整电路用电阻负载测试基本功能注意初次上电建议使用可调电源供电并串联电流表监视输入电流防止短路损坏元件。5.2 PID参数整定方法双闭环系统的参数整定应从内环电流环开始电流环整定先设Ki0, Kd0逐步增加Kp直到出现小幅振荡然后加入Ki消除静差最后加入Kd改善动态响应电压环整定电流环整定完成后固定其参数采用同样的方法整定电压环PID电压环带宽应低于电流环的1/5典型参数范围参考控制环Kp范围Ki范围Kd范围电流环0.1-1.00.01-0.10.001-0.01电压环0.5-5.00.1-1.00.01-0.1调试时可以先用阶跃响应观察系统行为给一个小的电压设定值变化如从5V到5.5V观察输出电压的响应曲线根据响应特性调整参数过冲大减小Kp或增大Kd响应慢增大Kp静差大增大Ki5.3 常见问题与解决方案输出电压振荡检查PCB布局功率回路是否过大尝试增加输出电容降低PID参数特别是Ki轻载时不稳定确认电感工作模式可能需要减小电感值调整最小占空比限制在DCM模式下可能需要不同的PID参数电流检测不准确检查采样电阻功率是否足够确认运放电路增益和偏置增加软件滤波如移动平均// 简单的软件滤波实现 #define FILTER_SIZE 5 float movingAverage(float newSample) { static float buffer[FILTER_SIZE] {0}; static uint8_t index 0; static float sum 0; sum - buffer[index]; buffer[index] newSample; sum newSample; index (index 1) % FILTER_SIZE; return sum / FILTER_SIZE; }6. 功能扩展与进阶优化基础功能实现后可以考虑以下增强功能提升电源的实用性和用户体验6.1 数字接口与远程控制UART/USB通信实现简单的命令行接口支持参数查询和设置添加数据记录功能void CLI_ProcessCommand(char* cmd) { if(strncmp(cmd, SET V , 6) 0) { float voltage atof(cmd 6); if(voltage 0 voltage VO_MAX) { vo_setpoint voltage; printf(OK Voltage set to %.2fV\n, voltage); } else { printf(ERROR Invalid voltage\n); } } // 其他命令处理... }OLED显示界面实时显示电压电流值参数设置菜单波形显示功能6.2 高级保护功能温度保护添加NTC测温根据温度调整最大输出电流过热关机保护软启动功能上电时逐步增加输出电压防止对容性负载的冲击电流void SoftStart(float targetVoltage) { const float step 0.1f; // 每步0.1V float current 0.0f; while(current targetVoltage) { current step; if(current targetVoltage) { current targetVoltage; } vo_setpoint current; HAL_Delay(10); // 每步10ms } }6.3 效率优化技巧同步整流用MOSFET替代续流二极管自适应死区控制根据电流调整死区时间动态频率调整轻载时降低开关频率经过这些优化我们的DIY数控电源不仅性能可媲美商业产品更关键的是通过这个项目我们深入理解了电力电子与嵌入式控制的精髓。在实际调试过程中示波器是最得力的工具建议重点关注开关节点波形、电感电流和输出电压纹波。
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2587810.html
如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!