PWM实现LED渐变效果及彩灯控制

简介: PWM实现LED渐变效果及彩灯控制

一、硬件

1. 核心电路设计

模块 参数要求 典型值
PWM控制器 带PWM输出的MCU STM32F103/ESP32
LED类型 共阳/共阴RGB LED WS2812B(数字控制)
限流电阻 根据LED正向压降计算 220Ω-1kΩ
电源 3.3V/5V系统供电 200mA以上

连接示例(STM32)

STM32引脚 → RGB LED
---------------------
TIM3_CH2 → R (PB5)
TIM3_CH3 → G (PB0)
TIM3_CH4 → B (PB1)
GND       → 公共地
3.3V      → 电源输入

2. 电气参数计算

  • 限流电阻计算

    download.png

(示例:V_CC=3.3V,V_f=2V,I=20mA → R=65Ω)

  • PWM频率选择

    download2.png

(推荐值:500Hz-2kHz)


二、软件

1. 基础渐变算法

// 定义颜色通道结构体
typedef struct {
   
    uint16_t duty;    // 占空比(0-65535)
    GPIO_TypeDef* port;
    uint16_t pin;
} LED_Channel;

// 线性渐变函数
void linear_fade(LED_Channel* channels, uint8_t num_channels, 
                 uint32_t target_duty, uint32_t step, 
                 uint32_t delay_ms) {
   
    static int16_t direction = 1;
    static uint32_t current_duty = 0;

    for(int i=0; i<num_channels; i++) {
   
        current_duty += step * direction;
        if(current_duty >= 65535 || current_duty <= 0) {
   
            direction *= -1;
            current_duty = constrain(current_duty, 0, 65535);
        }
        ledc_set_duty(LEDC_HIGH_SPEED_MODE, channels[i].channel, current_duty);
        ledc_update_duty(LEDC_HIGH_SPEED_MODE, channels[i].channel);
    }
    vTaskDelay(delay_ms / portTICK_PERIOD_MS);
}

2. 呼吸灯实现(STM32 HAL库)

// 定时器配置(TIM3)
void MX_TIM3_Init(void) {
   
    TIM_HandleTypeDef htim3;

    htim3.Instance = TIM3;
    htim3.Init.Prescaler = 72-1;    // 1MHz
    htim3.Init.CounterMode = TIM_COUNTERMODE_UP;
    htim3.Init.Period = 65535;      // 最大占空比
    htim3.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1;
    HAL_TIM_PWM_Init(&htim3);

    // 通道配置
    TIM_OC_InitTypeDef sConfigOC = {
   0};
    sConfigOC.OCMode = TIM_OCMODE_PWM1;
    sConfigOC.Pulse = 0;
    sConfigOC.OCPolarity = TIM_OCPOLARITY_HIGH;

    HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_2); // R
    HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_3); // G
    HAL_TIM_PWM_ConfigChannel(&htim3, &sConfigOC, TIM_CHANNEL_4); // B

    HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_2);
    HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_3);
    HAL_TIM_PWM_Start(&htim3, TIM_CHANNEL_4);
}

// 呼吸灯主循环
void breathing_led() {
   
    uint16_t duty = 0;
    uint8_t dir = 1;

    while(1) {
   
        __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_2, duty);
        __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_3, 65535-duty);
        __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_4, 0);

        duty += dir;
        if(duty == 0 || duty == 65535) dir = -dir;
        HAL_Delay(1);
    }
}

三、进阶功能实现

1. 彩虹渐变算法

// HSV转RGB算法
void hsv2rgb(float h, float s, float v, uint16_t* r, uint16_t* g, uint16_t* b) {
   
    float c = v * s;
    float x = c * (1 - fabs(fmod(h/60.0, 2) - 1));
    float m = v - c;

    if(h >= 0 && h < 60) {
    *r = c; *g = x; *b = 0; }
    else if(h >= 60 && h < 120) {
    *r = x; *g = c; *b = 0; }
    else if(h >= 120 && h < 180) {
    *r = 0; *g = c; *b = x; }
    else if(h >= 180 && h < 240) {
    *r = 0; *g = x; *b = c; }
    else if(h >= 240 && h < 300) {
    *r = x; *g = 0; *b = c; }
    else {
    *r = c; *g = 0; *b = x; }

    *r = (uint16_t)((*r + m) * 65535 / 255);
    *g = (uint16_t)((*g + m) * 65535 / 255);
    *b = (uint16_t)((*b + m) * 65535 / 255);
}

// 彩虹渐变主循环
void rainbow_fade() {
   
    float hue = 0.0;
    while(1) {
   
        hsv2rgb(hue, 1.0, 1.0, &r_duty, &g_duty, &b_duty);
        ledc_set_duty(LEDC_HIGH_SPEED_MODE, RED_CHANNEL, r_duty);
        ledc_set_duty(LEDC_HIGH_SPEED_MODE, GREEN_CHANNEL, g_duty);
        ledc_set_duty(LEDC_HIGH_SPEED_MODE, BLUE_CHANNEL, b_duty);
        ledc_update_duty(LEDC_HIGH_SPEED_MODE, RED_CHANNEL);
        ledc_update_duty(LEDC_HIGH_SPEED_MODE, GREEN_CHANNEL);
        ledc_update_duty(LEDC_HIGH_SPEED_MODE, BLUE_CHANNEL);

        hue += 0.5;
        if(hue >= 360) hue = 0;
        HAL_Delay(10);
    }
}

2. 呼吸灯优化方案

// 正弦波呼吸算法
void sine_wave_breathing() {
   
    float phase = 0.0;
    while(1) {
   
        float duty = (sin(phase * M_PI / 180) + 1) * 32767; // 16位范围
        __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_2, (uint32_t)duty);
        __HAL_TIM_SET_COMPARE(&htim3, TIM_CHANNEL_3, 65535 - (uint32_t)duty);
        phase += 1;
        if(phase >= 360) phase = 0;
        HAL_Delay(1);
    }
}

四、工程实现要点

1. 硬件保护电路

LED驱动电路设计
----------------
MCU PWM → 电阻 → 二极管 → 电感 → LED
                  ↑
                续流二极管

2. 中断服务程序

// 定时器中断处理
void TIM3_IRQHandler() {
   
    if(TIM_GetITStatus(TIM3, TIM_IT_UPDATE) != RESET) {
   
        TIM_ClearITPendingBit(TIM3, TIM_IT_UPDATE);

        // 动态调整占空比
        static uint16_t duty = 0;
        duty = (duty + 1024) % 65536;
        TIM_SetCompare2(TIM3, duty);
    }
}

3. 低功耗优化

// 进入睡眠模式
void enter_sleep_mode() {
   
    HAL_PWR_DisablePVD();
    HAL_PWR_EnterSLEEPMode(PWR_MAINREGULATOR_ON, PWR_SLEEPENTRY_WFI);
}

// 唤醒配置
void EXTI0_IRQHandler() {
   
    if(EXTI_GetITStatus(EXTI_Line0) != RESET) {
   
        EXTI_ClearITPendingBit(EXTI_Line0);
        HAL_PWR_DisableSleepDeep();
    }
}

参考代码 基于PWM的LED渐变效果及彩灯 www.youwenfan.com/contentalf/72808.html

五、扩展应用场景

1. 智能照明系统

// 光照感应控制
void ambient_light_control() {
   
    float lux = read_lux_sensor();
    float target_duty = 65535 * (lux / 10000.0); // 0-10000lux
    ledc_set_duty(LEDC_HIGH_SPEED_MODE, RED_CHANNEL, target_duty);
}

2. 节日灯光秀

// 预设灯光模式
typedef enum {
   
    MODE_STATIC,
    MODE_FADE,
    MODE_BREATHE,
    MODE_RAINBOW
} LightMode;

void execute_light_show() {
   
    static LightMode current_mode = MODE_STATIC;

    switch(current_mode) {
   
        case MODE_STATIC:
            set_static_color(0xFFFF, 0x0000, 0x0000); // 纯红色
            break;
        case MODE_FADE:
            linear_fade(channels, 3, 65535, 1024, 50);
            break;
        case MODE_BREATHE:
            sine_wave_breathing();
            break;
        case MODE_RAINBOW:
            rainbow_fade();
            break;
    }
}
相关文章
|
算法 搜索推荐 Android开发
android的A/B到底是什么?OTA升级又是什么?
android的A/B到底是什么?OTA升级又是什么?
787 0
|
存储 编解码 Linux
FFmpeg+SDL播放器开发实践:解析、解码、渲染全流程详解
FFmpeg+SDL播放器开发实践:解析、解码、渲染全流程详解
|
21天前
|
搜索推荐 数据可视化 数据库
用Python轻松打造专业PPT:自动化生成演示文稿全攻略
本文介绍如何用Python的python-pptx库自动化生成PPT,涵盖环境搭建、文本、图片、图表插入,以及批量生成与模板应用技巧。通过代码高效创建格式统一、内容丰富的演示文稿,大幅提升职场效率,适合报告、教学等场景,让PPT制作从繁琐变为智能。
407 1
|
21天前
|
分布式计算 监控 API
DMS Airflow:企业级数据工作流编排平台的专业实践
DMS Airflow 是基于 Apache Airflow 构建的企业级数据工作流编排平台,通过深度集成阿里云 DMS(Data Management Service)系统的各项能力,为数据团队提供了强大的工作流调度、监控和管理能力。本文将从 Airflow 的高级编排能力、DMS 集成的特殊能力,以及 DMS Airflow 的使用示例三个方面,全面介绍 DMS Airflow 的技术架构与实践应用。
|
Android开发
【Android 安装包优化】动态库打包配置 ( “armeabi-v7a“, “arm64-v8a“, “x86“, “x86_64“ APK 打包 CPU 指令集配置 | NDK 完整配置参考 )
【Android 安装包优化】动态库打包配置 ( “armeabi-v7a“, “arm64-v8a“, “x86“, “x86_64“ APK 打包 CPU 指令集配置 | NDK 完整配置参考 )
1815 0
【Android 安装包优化】动态库打包配置 ( “armeabi-v7a“, “arm64-v8a“, “x86“, “x86_64“ APK 打包 CPU 指令集配置 | NDK 完整配置参考 )
|
3月前
|
存储 Java 数据处理
Java映射操作:深入Map.getOrDefault与MapUtils方法
结合 `getOrDefault`方法的简洁性及 `MapUtils`的丰富功能,Java的映射操作变得既灵活又高效。合理地使用这些工具能够显著提高数据处理的速度和质量。开发人员可以根据具体的应用场景选择适宜的方法,以求在性能和可读性之间找到最佳平衡。
176 0
|
流计算
(文章复现)基于电力系统碳排放流理论的碳排放分摊模型研究
之前写过一篇博客复现论文《电力系统碳排放流的计算方法初探》,那篇文章模型比较简单,没有考虑网损。 今天要复现的这篇文献,考虑了功率损耗,将碳排放在用电负荷、网络损耗以及厂用电之间进行分配,具体原理如下。
|
Java Android开发
解决Android编译报错:Unable to make field private final java.lang.String java.io.File.path accessible
解决Android编译报错:Unable to make field private final java.lang.String java.io.File.path accessible
3274 1
|
人工智能 自然语言处理 开发者
ChatGPT在国内的使用限制,国内的ChatGPT替代工具
ChatGPT在国内的使用限制,国内的ChatGPT替代工具
368 0