PWM(脉冲宽度调制)是嵌入式开发中最常用的信号控制技术之一。简单来说,PWM就是通过快速开关电源来控制平均电压的技术。想象一下你用手指快速开关水龙头,开关速度足够快时,水流看起来就像是稳定的一部分流量——这就是PWM的工作原理。
在ESP32上,LEDC(LED PWM控制器)是专门为LED调光设计的硬件PWM模块,但它实际上可以用于各种需要PWM信号的场景。LEDC有16个独立通道,分为两组:
我经常建议新手从高速通道开始使用,因为它们能提供更高的频率和更稳定的信号。在实际项目中,我发现通道0-3通常被保留用于特殊功能,所以从通道4开始使用是个好习惯。
让我们从最简单的呼吸灯开始。你只需要:
连接方式很简单:LED正极接ESP32的GPIO(比如GPIO2),负极通过电阻接地。这里我强烈建议使用限流电阻,即使ESP32的GPIO有短路保护,直接连接LED也容易损坏引脚。
配置LEDC通道有三个关键参数:
cpp复制ledcSetup(channel, freq, resolution_bits);
对于呼吸灯,我推荐以下配置:
cpp复制ledcSetup(0, 5000, 8); // 通道0,5kHz频率,8位分辨率
呼吸灯的核心是让亮度平滑变化。这里有个小技巧:不要线性改变占空比,而是采用类似正弦曲线的变化,这样看起来更自然。下面是我优化过的呼吸灯代码:
cpp复制int breatheTime = 3; // 呼吸周期3秒
int steps = 100; // 分100步完成呼吸
void setup() {
ledcSetup(0, 5000, 8);
ledcAttachPin(2, 0);
}
void loop() {
for(int i=0; i<steps; i++){
// 使用正弦曲线计算亮度
float rad = PI * i / steps;
int duty = (int)(127.5 * (sin(rad - PI/2) + 1));
ledcWrite(0, duty);
delay(breatheTime * 1000 / steps);
}
}
这个实现比简单的线性变化看起来舒服多了,我在多个项目中都采用了这种算法。
控制电机比控制LED复杂得多,主要区别在于:
我曾在项目中因为没有考虑这些因素烧毁过MOSFET,所以特别提醒:一定要使用合适的驱动电路!最简单的方案是使用带保护功能的电机驱动模块,比如L298N或DRV8871。
电机控制需要不同的PWM参数:
cpp复制// 直流有刷电机典型配置
ledcSetup(1, 20000, 10); // 通道1,20kHz频率,10位分辨率
为什么选择20kHz?因为:
下面是一个带缓启动功能的电机控制代码,可以有效减少启动时的电流冲击:
cpp复制#define MOTOR_PIN 13
#define MOTOR_CH 1
void setup() {
ledcSetup(MOTOR_CH, 20000, 10);
ledcAttachPin(MOTOR_PIN, MOTOR_CH);
}
void setMotorSpeed(int speed) { // speed: 0-1023
static int currentSpeed = 0;
// 缓加速/减速
while(currentSpeed != speed) {
if(currentSpeed < speed) currentSpeed++;
else currentSpeed--;
ledcWrite(MOTOR_CH, currentSpeed);
delay(5); // 调整这个值改变加速/减速时间
}
}
void loop() {
// 从0加速到全速
setMotorSpeed(1023);
delay(2000);
// 减速到停止
setMotorSpeed(0);
delay(2000);
}
有时我们需要多个PWM通道同步变化,比如RGB LED或双电机控制。ESP32的LEDC有个特性:相同定时器的通道会同步更新。这意味着我们可以这样配置:
cpp复制// 使用同一个定时器的不同通道
ledcSetup(0, 5000, 8); // 定时器0
ledcSetup(1, 5000, 8); // 定时器0
ledcSetup(2, 5000, 8); // 定时器0
这样调用ledcWrite()时,三个通道会同时更新,避免颜色或速度变化不同步的问题。
PWM信号不稳定
电机响应迟钝
LED闪烁或不均匀
我在一个四轴飞行器项目中遇到过PWM信号抖动的问题,最后发现是因为WiFi和蓝牙的射频干扰。解决方案是: