当前位置: 首页 > article >正文

单片机裸机编程-时机管理

对于 RTOS 实时操作系统,我们是通过 TASK(任务)进行底层操作的,这与裸机编程中的函数(fun)类似。不同的任务或函数实现不同的功能,在RTOS中,单片机有信号量、队列等不同任务之间的通信机制,但对于裸机编程来说就引出了一个问题:不同任务或函数是如何配合工作的呢?换句话说,单片机是如何知道什么时候该做什么事情,或者为什么做完一件事之后再做另一件事呢?


1、裸机的逻辑轮询

在裸机编程中,任务的执行顺序通常是固定的,由程序的流程控制语句(如 ifwhileswitch 等)决定程序从入口点开始,按顺序执行代码,直到遇到分支或循环。当然,也可以通过中断来实现某些功能。

例如,以下代码展示了通过全局变量的状态来控制不同函数的执行逻辑:

void main() {
    while (1) {
        if (sensor_data_ready) {  // 检查传感器数据是否准备好
            process_sensor_data();
        }
        if (button_pressed) {     // 检查按钮是否被按下
            handle_button_press();
        }
        update_display();         // 更新显示
    }
}

在这个例子中,程序通过全局变量(sensor_data_readybutton_pressed)来判断是否执行某个函数。

也就是说,在不同的执行函数之间的通信使用的是全局变量,或者说是标志位。我们通过if,switch这样的逻辑语句让单片机知道在什么情况下该做什么事,所以只需要一直轮询下去即可

这种方式简单直接,但存在以下问题:

  1. 耦合性高:全局变量的使用使得代码之间的耦合性增加,难以维护和扩展。

  2. 灵活性差:任务的执行顺序固定,难以动态调整。


2、使用状态机进行任务管理

为了改善上述问题,我们可以引入 状态机 的概念,对不同的任务进行局部管理。状态机通过定义不同的状态和状态之间的转换条件,使得代码更加模块化和灵活。例如:

typedef enum {
    STATE_IDLE,
    STATE_PROCESS_SENSOR,
    STATE_HANDLE_BUTTON,
    STATE_UPDATE_DISPLAY
} StateTypeDef;

StateTypeDef currentState = STATE_IDLE;

void main() {
    while (1) {
        switch (currentState) {
            case STATE_IDLE:
                if (sensor_data_ready) {
                    currentState = STATE_PROCESS_SENSOR;
                } else if (button_pressed) {
                    currentState = STATE_HANDLE_BUTTON;
                }
                break;

            case STATE_PROCESS_SENSOR:
                process_sensor_data();
                currentState = STATE_IDLE;
                break;

            case STATE_HANDLE_BUTTON:
                handle_button_press();
                currentState = STATE_IDLE;
                break;

            case STATE_UPDATE_DISPLAY:
                update_display();
                currentState = STATE_IDLE;
                break;
        }
    }
}

通过状态机,我们可以清晰地定义每个任务的执行条件和状态转换逻辑,从而提高代码的可读性和可维护性。


3、在裸机中实现时间片轮询

进一步思考,我们可以在裸机编程中借鉴 RTOS 的时间片轮询机制。虽然裸机没有 RTOS 内核的支持,但可以通过定时器中断来实现类似的效果。例如:

  1. 设置定时器中断:配置定时器以固定频率(如 10ms)触发中断。

  2. 维护任务状态:在定时器中断中维护一个任务状态数组,记录每个任务的执行状态和剩余时间片。

  3. 轮询任务执行:在主循环中,根据任务状态数组依次执行每个任务的一部分。

示例代码如下:

#define TASK_COUNT 3
#define TIME_QUANTUM 10  // 时间片大小,单位为毫秒

typedef struct {
    void (*taskFunc)(void);  // 任务函数指针
    int remainingTime;       // 剩余时间片
} TaskTypeDef;

TaskTypeDef tasks[TASK_COUNT] = {
    {process_sensor_data, TIME_QUANTUM},
    {handle_button_press, TIME_QUANTUM},
    {update_display, TIME_QUANTUM}
};

void Timer_ISR(void) {
    static int tick = 0;
    tick++;
}

void main() {
    int currentTask = 0;

    while (1) {
        if (tasks[currentTask].remainingTime > 0) {
            tasks[currentTask].taskFunc();  // 执行当前任务
            tasks[currentTask].remainingTime--;
        }

        currentTask = (currentTask + 1) % TASK_COUNT;  // 轮询下一个任务
    }
}

通过这种方式,我们可以在裸机中实现类似 RTOS 的时间片轮询机制,使得任务的执行更加公平和灵活。

个人喜欢使用状态机进行裸机编程,这样直接是多个代码块直接的裸机判断。也就相当于是有多个while(1)循环,方便代码的管理和调试。在不同的state模式下执行不同的小模块代码。


http://www.kler.cn/a/560920.html

相关文章:

  • 电机控制的空间矢量调制 (SVPWM)
  • ubuntu-24.04.1-desktop 中的 QT6.7 QtCreator 调试程序
  • 数据结构:双链表list
  • elasticsearch在windows上的配置
  • BUUCTF--[极客大挑战 2019]RCE ME
  • 【深度学习】强化学习(RL)-A3C(Asynchronous Advantage Actor-Critic)
  • [深度学习][python]yolov12+bytetrack+pyqt5实现目标追踪
  • linux驱动编程配置----(二)
  • 论文笔记(七十二)Reward Centering(五)
  • 【Gee】Day1:HTTP 基础
  • 深入讲解微信小程序 <canvas> 标签的 type=“2d“属性
  • 漏洞文字版表述一句话版本(漏洞危害以及修复建议),通常用于漏洞通报中简洁干练【持续更新中】
  • 【Java项目】基于SpringBoot的【旅游管理系统】
  • Python爬虫系列教程之第十五篇:爬取电商网站商品信息与数据分析
  • C++ QT 6.6.1 QCustomPlot的导入及使用注意事项和示例 | 关于高版本QT使用QCustomPlot报错问题解决的办法
  • SSL和TLS:深入了解网络安全的基石
  • 【DeepSeek与鸿蒙HarmonyOS:开启应用开发新次元】
  • DSP芯片C6678的SRIO及其中断跳转的配置
  • LeetCode 热题 100 94. 二叉树的中序遍历
  • 基于SpringBoot的“流浪动物救助系统”的设计与实现(源码+数据库+文档+PPT)