状态机实现RGB灯跳变

1.项目功能梗概

因为原本使用的为for循环进行遍历,然后依次执行代码,但是由于看门狗的存在,不能使用delay_ms这种死延时。所以现在打算定时器回调函数控制状态机状态这种方法。

2.状态机

作用

当系统需要执行某个任务时,可以根据状态机的状态选择不同的操作,例如:

  • 控制系统状态:例如控制器根据状态机的状态选择执行不同的操作,从而控制整个系统的状态,例如开关灯、控制电机等。
  • 系统调度:例如操作系统根据状态机的状态选择执行不同的任务,从而实现系统的调度。
  • 系统事件处理:例如网络通信系统根据状态机的状态选择不同的数据处理方式,例如处理接收到的数据、发送数据等。
  • 系统错误处理:例如系统出现错误时,可以根据状态机的状态选择不同的错误处理方式,例如重新启动系统、输出错误信息等。

总之,状态机可以帮助我们将系统的复杂性分解为多个简单的状态,根据不同的状态选择执行不同的操作,从而更好地实现系统的设计与开发。

主要流程

  1. 定义状态和事件

    首先需要定义系统可能存在的所有状态,通常用枚举类型来表示。同时需要定义可能发生的所有事件,例如输入的数据、定时器到达等。

  2. 初始化状态机

    在程序启动时,需要将状态机初始化为特定的状态。

  3. 定义状态转换和动作

    使用switch-case if-else 函数指针等语句来定义状态转换和动作。当事件发生时,判断当前状态,并根据不同的事件执行相应的动作,并将状态转换为下一个状态。

  4. 事件处理

    当有事件发生时,将事件作为参数传递给状态机,并调用状态转换和动作函数。

  5. 循环执行

    状态机通常作为一个独立的任务运行,并在一个无限循环中等待事件的发生。

3.实现过程相关思考

说实话,尝试了下突然有用标志位的思路了


// 定义全局指针变量
u8 *global_bt_receive_buffer_gradually;
u32 global_length_gradually;
volatile u8 time_flag_gradually = 0;
void handleColor_arrayChange_gradually(u8 *bt_receive_buffer, u32 length)
{



    printf("通过数组控制灯颜色状态\n");
    static int j = 0;
    static int i = 5;
    u8 STEPS = 30;
    static u8 i_balance = 3;
         // 获取初始亮度i-
    u8 initial_red_value = bt_receive_buffer[i];
    u8 initial_green_value = bt_receive_buffer[i+1];
    u8 initial_blue_value = bt_receive_buffer[i+2];

    for (; (i < (length - 5)) && (time_flag_gradually == 0);)
    {

        u8 target_red_value = bt_receive_buffer[i + 3];
        u8 target_green_value = bt_receive_buffer[i + 4];
        u8 target_blue_value = bt_receive_buffer[i + 5];
        printf("i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------%d\n", i);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_red_value);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_green_value);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_blue_value);
        // 计算每个颜色通道的增量
        int red_increment = (target_red_value - initial_red_value) / STEPS;
        int green_increment = (target_green_value - initial_green_value) / STEPS;
        int blue_increment = (target_blue_value - initial_blue_value) / STEPS;
        // 逐步更新颜色值,使其渐变到目标值
        for (; (j < STEPS) && (time_flag_gradually == 0); j++)
        {

            u8 current_red_value = initial_red_value + (red_increment * j);
            u8 current_green_value = initial_green_value + (green_increment * j);
            u8 current_blue_value = initial_blue_value + (blue_increment * j);

            handleColorChange(current_red_value, current_green_value, current_blue_value);

            // delay_2ms(24); // 延时一段时间,控制灯的变化速度
            printf("initial_red_value----initial_red_value----initial_red_value----initial_red_value----initial_red_value----%d\n", initial_red_value);
            printf("initial_green_value-----initial_green_value-------initial_green_value-------initial_green_value-------%d\n", initial_green_value);
            printf("initial_blue_valueinitial_blue_valueinitial_blue_valueinitial_blue_valueinitial_blue_valueinitial_b%d\n", initial_blue_value);
            printf("红色值每一次增量------红色值每一次增量--------红色值每一次增量--------红色值每一次增量-------红色值每一次增量%d\n", red_increment);
            printf("绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------%d\n", green_increment);
            printf("蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------%d\n", blue_increment);
            time_flag_gradually = 1;
            printf("jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj===================%d\n", j);
        }
        if (j >= STEPS)
        {

            j = 0;
            i = i + 3;
        }

        printf("第一组 RGB 值已经渐变到第二组一样\n");

        // 更新初始亮度为目标亮度,以便下一组渐变
        initial_red_value = target_red_value;
        initial_green_value = target_green_value;
        initial_blue_value = target_blue_value;
    }
    if (i >= (length - 5))
    {
        i = 5;
    }

    global_bt_receive_buffer_gradually = bt_receive_buffer;
    global_length_gradually = length;
}
u32 timeout_msec_gradually = 5; // 设置定时时间为10毫秒
void timeout_callback_gradually(void *priv)
{
    // 每10ms
    static u8 i = 5;
    // printf("回调函数渐变\n");
    printf("回调函数渐变%d\n", time_flag_gradually);
    // printf("time_flag_gradually != 0time_flag_gradually != 0time_flag_gradually != 0time_flag_gradually != 0time_flag_gradually != 0\n");
    i = i + 5;
    if (i == 50)
    {
        time_flag_gradually = 0;
        i = 0;
    }
    handleColor_arrayChange_gradually(global_bt_receive_buffer_gradually, global_length_gradually);
}

其实就是把第一个for循环里的数组移位转移到了小的for循环里,只有当小的for循环里面已经将颜色渐变全部实现了,才允许i++实现移位。不过有个问题就是颜色渐变完成后,会从最后一个颜色突变到第一个颜色。这里应该是在设置代码初始化时的重新赋值的原因,导致缺失了一段颜色变化。

 

  • 将所要的颜色数据数据提取出来,作为一个数组。每一次颜色渐变完成,就将数组的后三位赋值到前三位,以此类推,重新执行函数。
  • 第二种,写死,每次多执行一段最后一个颜色渐变到第一个颜色的代码。

疑惑,有没有使用更简洁的方法,或者说环形数组对这种循环显示的是不是更好。

贴一下目前代码图片

u16 timer_id_color_change_suddenly = 0;
u16 timer_id_color_change_gradually = 0;
void *private_data = NULL; // 设置私有参数为NULL
u32 timeout_msec = 10;     // 设置定时时间为10毫秒
u8 priority = 1;           // 设置优先级为1
void timeout_callback(void *priv)
{

    static u16 i = 0;
    printf("回调函数跳变\n");

    i = i + 10;
    if (i == 1000)
    {
        printf("ifififiifiifififiififififi\n");
        time_flag_suddenly = 1;
        i = 0;
    }
    handleColor_arrayChange_suddenly(global_bt_receive_buffer, global_length, global_color_mode_record, global_color1);
}

// 定义全局指针变量
u8 *global_bt_receive_buffer_gradually;
u32 global_length_gradually;
volatile u8 time_flag_gradually = 0;
void handleColor_arrayChange_gradually(u8 *bt_receive_buffer, u32 length)
{

    printf("通过数组控制灯颜色状态\n");
    static int j = 0;
    static int i = 5;
    u8 STEPS = 30;
    static u8 i_balance = 3;
    static flag = 1;
    // 获取初始亮度i-
    u8 initial_red_value = bt_receive_buffer[i];
    u8 initial_green_value = bt_receive_buffer[i + 1];
    u8 initial_blue_value = bt_receive_buffer[i + 2];

    for (; (i < (length - 5)) && (time_flag_gradually == 0);)
    {

        u8 target_red_value = bt_receive_buffer[i + 3];
        u8 target_green_value = bt_receive_buffer[i + 4];
        u8 target_blue_value = bt_receive_buffer[i + 5];
        printf("i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------i的值为-----------%d\n", i);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_red_value);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_green_value);
        printf("target_red_value------------------target_red_value---------------target_red_value%d\n", target_blue_value);
        // 计算每个颜色通道的增量
        int red_increment = (target_red_value - initial_red_value) / STEPS;
        int green_increment = (target_green_value - initial_green_value) / STEPS;
        int blue_increment = (target_blue_value - initial_blue_value) / STEPS;
        // 逐步更新颜色值,使其渐变到目标值
        for (; (j < STEPS) && (time_flag_gradually == 0); j++)
        {

            u8 current_red_value = initial_red_value + (red_increment * j);
            u8 current_green_value = initial_green_value + (green_increment * j);
            u8 current_blue_value = initial_blue_value + (blue_increment * j);

            handleColorChange(current_red_value, current_green_value, current_blue_value);

            // delay_2ms(24); // 延时一段时间,控制灯的变化速度
            printf("initial_red_value----initial_red_value----initial_red_value----initial_red_value----initial_red_value----%d\n", initial_red_value);
            printf("initial_green_value-----initial_green_value-------initial_green_value-------initial_green_value-------%d\n", initial_green_value);
            printf("initial_blue_valueinitial_blue_valueinitial_blue_valueinitial_blue_valueinitial_blue_valueinitial_b%d\n", initial_blue_value);
            printf("红色值每一次增量------红色值每一次增量--------红色值每一次增量--------红色值每一次增量-------红色值每一次增量%d\n", red_increment);
            printf("绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------绿色值每一次增量------%d\n", green_increment);
            printf("蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------蓝色值每一次增量------%d\n", blue_increment);
            time_flag_gradually = 1;
            printf("jjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjjj===================%d\n", j);
        }
        if (j >= STEPS)
        {

            j = 0;
            i = i + 3;
        }

        printf("第一组 RGB 值已经渐变到第二组一样\n");

        // 更新初始亮度为目标亮度,以便下一组渐变
        initial_red_value = target_red_value;
        initial_green_value = target_green_value;
        initial_blue_value = target_blue_value;
    }
    if (i >= (length - 5))
    {
        
        i=5;

    }

    global_bt_receive_buffer_gradually = bt_receive_buffer;
    global_length_gradually = length;
}

        

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/130672.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

力扣字符串--总结篇

前言 字符串学了三天&#xff0c;七道题。初窥kmp&#xff0c;已经感受到算法的博大精深了。 内容 对字符串的操作可以归结为以下几类&#xff1a; 字符串的比较、连接操作&#xff08;不同编程语言实现方式有所不同&#xff09;&#xff1b; 涉及子串的操作&#xff0c;比…

Python数据结构: 列表(List)详解

在Python中&#xff0c;列表&#xff08;List&#xff09;是一种有序、可变的数据类型&#xff0c;被广泛用于存储和处理多个元素。列表是一种容器&#xff0c;可以包含任意数据类型的元素&#xff0c;包括数字、字符串、列表、字典等。本文将深入讨论列表的各个方面&#xff0…

strcat()用法

描述 头文件&#xff1a;<string.h>char *strcat&#xff08;char *dest&#xff0c; const char *src&#xff09;功能&#xff1a;将src字符串加到dest上&#xff0c;并返回指向dest字符串的指针。 举例 #include<stdio.h> #include<string.h> int mai…

基恩士软件的基本操作(一)

今天就来学习基恩士软件的基础操作&#xff0c;欢迎大家的指正&#xff01;&#xff01;&#xff01; 基本操作 KV STUDIO 基恩士编程软件的名称就KV STUDIO。安装软件地址KV STUDIO的安装与实践 项目的创建 1&#xff0c;双击KV STUDIO. 2&#xff0c;新建项目 单元编辑器…

【MATLAB源码-第74期】基于matlab的OFDM-IM索引调制系统不同频偏误码率对比,对比OFDM系统。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 OFDM-IM索引调制技术是一种新型的无线通信技术&#xff0c;它将正交频分复用&#xff08;OFDM&#xff09;和索引调制&#xff08;IM&#xff09;相结合&#xff0c;以提高频谱效率和系统容量。OFDM-IM索引调制技术的基本思想…

【docker:容器提交成镜像】

容器创建部分请看&#xff1a;点击此处查看我的另一篇文章 容器提交为镜像 docker commit -a "sinwa lee" -m "首页变化" mynginx lxhnginx:1.0docker run -d -p 88:80 --name lxhnginx lxhnginx:1.0为啥没有变啊&#xff0c;首页&#xff1f; 镜像打包 …

SMART PLC模拟量上下限报警功能块(梯形图代码)

博途PLC模拟量偏差报警功能块请参考下面的文章链接: 模拟量偏差报警功能块(SCL代码)_RXXW_Dor的博客-CSDN博客文章浏览阅读594次。工业模拟量采集的相关基础知识,可以查看专栏的系列文章,这里不再赘述,常用链接如下:PLC模拟量采集算法数学基础(线性传感器)_plc傳感器數…

ElasticSearch7.x - HTTP 操作 - 文档操作

创建文档(添加数据) 索引已经创建好了,接下来我们来创建文档,并添加数据。这里的文档可以类比为关系型数 据库中的表数据,添加的数据格式为 JSON 格式 向 ES 服务器发 POST 请求 :http://192.168.254.101:9200/shopping/_doc 请求体内容为: {"title":"小…

多维时序 | MATLAB实现SOM-BP自组织映射结合BP神经网络的多变量时间序列预测

多维时序 | MATLAB实现SOM-BP自组织映射结合BP神经网络的多变量时间序列预测 目录 多维时序 | MATLAB实现SOM-BP自组织映射结合BP神经网络的多变量时间序列预测预测效果基本介绍模型描述程序设计参考资料 预测效果 基本介绍 MATLAB实现SOM-BP自组织映射结合BP神经网络的多变量时…

Dell戴尔灵越Inspiron 7700 AIO一体机电脑原厂预装Windows10系统

链接&#xff1a;https://pan.baidu.com/s/1-slgR9t4Df_eko0Y6xaeyw?pwdmk0p 提取码&#xff1a;mk0p 灵越7700一体机原装出厂系统自带声卡驱动、无线网卡驱动、面部识别等所有驱动、出厂主题壁纸、系统属性专属LOGO标志、Office办公软件、MyDell等预装程序 由于时间关系,…

软件测试项目实战经验附视频以及源码【商城项目,app项目,电商项目,银行项目,医药项目,金融项目】(web+app+h5+小程序)

前言&#xff1a; ​​大家好&#xff0c;我是阿里测试君。 最近很多小伙伴都在面试&#xff0c;但是对于自己的项目经验比较缺少。阿里测试君再度出马&#xff0c;给大家找了一个非常适合练手的软件测试项目&#xff0c;此项目涵盖web端、app端、h5端、小程序端&#xff0c;…

【Linux奇遇记】我和Linux的初次相遇

&#x1f308;个人主页: Aileen_0v0 &#x1f525;系列专栏:Linux奇遇记系列专栏&#x1f4ab;"没有罗马,那就自己创造罗马~" 目录 前端和后端的介绍 1.前端 2.后端 3.前后端区别 Linux在前后端开发中的角色 如何学习Linux 去进行程序开发 Linux的常见根目…

[工业自动化-10]:西门子S7-15xxx编程 - PLC主站 - 信号量:数字量

目录 前言&#xff1a; 一、工业现场常见信号的分类 二、IO数字量模块 2.1 概述 2.2 PLC的数字量是24V还是5V电压&#xff1f; 2.2 数字量模块的安装与接线 2.3 数字量模的注意事项 前言&#xff1a; 一、工业现场常见信号的分类 在工业自动化领域&#xff0c;常常需要使…

【word技巧】word文件中的图片,批量提取

如果你需要word文件中的图片做其他事情&#xff0c;除了一张张的进行图片另存为以外&#xff0c;我们还有其他方法可以批量一次性保存word文档中的图片嘛&#xff1f;今天分享两个方法&#xff0c;批量保存word文档图片。 方法一&#xff1a; 将文件进行另存为&#xff0c;在…

OpenWRT配置SFTP远程文件传输,让数据分享更安全

文章目录 前言 1. openssh-sftp-server 安装2. 安装cpolar工具3.配置SFTP远程访问4.固定远程连接地址 前言 本次教程我们将在OpenWRT上安装SFTP服务&#xff0c;并结合cpolar内网穿透&#xff0c;创建安全隧道映射22端口&#xff0c;实现在公网环境下远程OpenWRT SFTP&#xf…

vue项目pdf文件的预览

1.下载 您可以在以下网址下载pdfjsLib&#xff1a;https://github.com/mozilla/pdf.js pdfjsLib是一个开源项目&#xff0c;您可以在GitHub上找到其源代码和相关资源。 2.放置文件位置 3.进入 在index.html引入 <script src"<% BASE_URL %>static/pdfjs-dist/b…

Linux - 基础IO(Linux 当中的文件,文件系统调用接口,文件描述符)- 上篇

前言 首先&#xff0c;关于文件我们最先要理解的是&#xff0c;文件不仅仅存储的是数据&#xff0c;一个文件包括 内容 数据。内容好理解&#xff0c;就是我们先要这文件存储哪一些数据&#xff0c;这些数据就是文件的内容。 但是&#xff0c;在计算机当中&#xff0c;有两种…

Windows 10 下使用Visual Studio 2017 编译CEF SDK

1.下载CEF SDK 由于需要跑在32位的机器&#xff0c;所以选择下载32位的SDKCEF Automated Builds 选择 Current Stable Build (Preferred) &#xff0c;这是当前稳定版本&#xff0c;CEF版本118 下载成功解压 2.下载编译工具 CMake 下载地址&#xff1a;CMake 配置CMake指向…

n-gram语言模型——句子概率分布计算与平滑

n-gram语言模型——句子概率分布计算与平滑 前言 语言模型 等价假设 n元语法 句子概率分布计算方式 数据平滑 Lidstone平滑(1-gram) Laplace平滑(1-gram) 附上两种平滑在1-gram下代码 Lidstone平滑与Laplace平滑(2-gram) 附上两种平滑在2-gram下代码 前言 语言模型…

黑窗口连接远程服务

ssh root192.168.x.x 回车输入密码 查看docker docker ps 停止正在运行的服务 docker stop xxxxx 删除服务 docker rm xxxxx 查看镜像 docker images 删除镜像 docker rmi xxxxx 删除镜像 启动并运行整个服务 docker compose up -d jar包名称 idea 使用tcp方式连接docker 配置d…