news 2026/5/6 9:24:41

用STM32和PID算法做个数控电源:从BUCK电路到双闭环控制的完整实战

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
用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布局:

  1. 功率回路最小化:缩短MOSFET、电感和二极管之间的连线
  2. 地平面分割:将功率地和信号地分开,单点连接
  3. 退耦电容:在MCU和运放电源引脚就近放置100nF陶瓷电容
  4. 散热设计:为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采样配置

电压电流的精确采样是闭环控制的基础。需要注意:

  1. 使用定时器触发ADC,实现固定频率采样
  2. 配置DMA传输采样结果,减少CPU开销
  3. 合理设置采样保持时间,确保精度
// 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 控制状态机实现

数控电源需要根据工作状态切换控制模式,典型的状态包括:

  1. 恒压模式(CV):优先稳定输出电压
  2. 恒流模式(CC):当电流达到设定值时切换
  3. 保护模式:检测到异常时关闭输出
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 开环测试流程

  1. PWM测试:不接功率电路,用示波器确认PWM波形正常
  2. 驱动测试:接MOSFET但不接电感,检查栅极驱动波形
  3. 轻载测试:接完整电路,用电阻负载测试基本功能

注意:初次上电建议使用可调电源供电,并串联电流表监视输入电流,防止短路损坏元件。

5.2 PID参数整定方法

双闭环系统的参数整定应从内环(电流环)开始:

  1. 电流环整定

    • 先设Ki=0, Kd=0
    • 逐步增加Kp直到出现小幅振荡
    • 然后加入Ki消除静差
    • 最后加入Kd改善动态响应
  2. 电压环整定

    • 电流环整定完成后,固定其参数
    • 采用同样的方法整定电压环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

调试时可以先用阶跃响应观察系统行为:

  1. 给一个小的电压设定值变化(如从5V到5.5V)
  2. 观察输出电压的响应曲线
  3. 根据响应特性调整参数:
    • 过冲大:减小Kp或增大Kd
    • 响应慢:增大Kp
    • 静差大:增大Ki

5.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 数字接口与远程控制

  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"); } } // 其他命令处理... }
  1. OLED显示界面
    • 实时显示电压电流值
    • 参数设置菜单
    • 波形显示功能

6.2 高级保护功能

  1. 温度保护

    • 添加NTC测温
    • 根据温度调整最大输出电流
    • 过热关机保护
  2. 软启动功能

    • 上电时逐步增加输出电压
    • 防止对容性负载的冲击电流
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 效率优化技巧

  1. 同步整流:用MOSFET替代续流二极管
  2. 自适应死区控制:根据电流调整死区时间
  3. 动态频率调整:轻载时降低开关频率

经过这些优化,我们的DIY数控电源不仅性能可媲美商业产品,更关键的是通过这个项目,我们深入理解了电力电子与嵌入式控制的精髓。在实际调试过程中,示波器是最得力的工具,建议重点关注开关节点波形、电感电流和输出电压纹波。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/6 9:24:38

VGG-T3:线性复杂度3D重建技术解析与实践

1. 项目概述VGG-T3是一种突破性的离线3D重建方法&#xff0c;它通过创新的网络架构设计&#xff0c;将传统3D重建的计算复杂度从二次方降低到线性级别。这个由牛津大学视觉几何组&#xff08;VGG&#xff09;开发的技术&#xff0c;正在改变大规模场景重建的游戏规则。在实际项…

作者头像 李华
网站建设 2026/5/6 9:24:06

用HAL库GPIO中断实现按键长按、短按与连发:STM32F103C8T6状态机实战

基于HAL库的STM32智能按键状态机设计与实战 在嵌入式设备开发中&#xff0c;按键作为最基础的人机交互接口&#xff0c;其处理逻辑的优劣直接影响用户体验。传统简单的延时消抖和电平检测已无法满足现代智能设备对按键交互的需求——我们需要能够准确识别单击、双击、长按甚至连…

作者头像 李华
网站建设 2026/5/6 9:24:04

STM32F103驱动TM7711 24位ADC芯片,从电路设计到代码调试的完整避坑指南

STM32F103驱动TM7711 24位ADC芯片实战&#xff1a;从硬件设计到软件调优的全流程解析 在嵌入式系统开发中&#xff0c;高精度模拟信号采集一直是设计难点。TM7711作为一款国产24位Σ-Δ型ADC芯片&#xff0c;以不到HX711一半的价格实现了同等精度的模数转换&#xff0c;特别适合…

作者头像 李华
网站建设 2026/5/6 9:23:35

如何用QMCDecode轻松解锁QQ音乐加密音频:Mac用户的终极解决方案

如何用QMCDecode轻松解锁QQ音乐加密音频&#xff1a;Mac用户的终极解决方案 【免费下载链接】QMCDecode QQ音乐QMC格式转换为普通格式(qmcflac转flac&#xff0c;qmc0,qmc3转mp3, mflac,mflac0等转flac)&#xff0c;仅支持macOS&#xff0c;可自动识别到QQ音乐下载目录&#xf…

作者头像 李华
网站建设 2026/5/6 9:10:29

解锁游戏无限可能:MelonLoader模组加载器完全指南

解锁游戏无限可能&#xff1a;MelonLoader模组加载器完全指南 【免费下载链接】MelonLoader The Worlds First Universal Mod Loader for Unity Games compatible with both Il2Cpp and Mono 项目地址: https://gitcode.com/gh_mirrors/me/MelonLoader 你是否曾经想过为…

作者头像 李华