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

单片机裸机编程:状态机与其他高效编程框架

在单片机裸机编程中,状态机是一种非常强大的工具,能够有效管理复杂的逻辑和任务切换。除了状态机,还有其他几种编程模式可以在不使用 RTOS 的情况下实现高效的程序设计。以下是一些常见的方法:

1. 状态机编程

状态机通过定义系统的不同状态和状态之间的转换规则,将复杂的逻辑分解为简单的状态和事件处理。它适用于事件驱动的系统,能够有效管理任务切换和逻辑复杂性。

实现思路:
  • 定义状态枚举类型。

  • 使用状态变量记录当前状态。

  • 在主循环中根据当前状态执行对应的任务。

  • 根据事件或条件触发状态转换。

示例代码:

typedef enum {STATE_IDLE,STATE_PROCESS_SENSOR,STATE_HANDLE_BUTTON,STATE_UPDATE_DISPLAY
} StateTypeDef;StateTypeDef currentState = STATE_IDLE;void process_sensor_data(void) {// 处理传感器数据
}void handle_button_press(void) {// 处理按钮事件
}void update_display(void) {// 更新显示
}void main(void) {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;}}
}

2. 时间片轮询(Super Loop + 定时器)

时间片轮询是一种模拟多任务调度的方法,通过定时器中断实现时间片的管理。每个任务被分配一个固定的时间片,在主循环中依次执行各个任务的一部分。当时间片用完时,切换到下一个任务。

实现思路:
  • 设置一个定时器中断,用于记录时间片的结束。

  • 在主循环中,根据时间片的计数器决定当前任务是否继续执行。

示例代码:

#define TASK_COUNT 3
#define TIME_QUANTUM 10  // 时间片大小,单位为毫秒typedef struct {void (*taskFunc)(void);  // 任务函数指针int remainingTime;       // 剩余时间片
} TaskTypeDef;TaskTypeDef tasks[TASK_COUNT] = {{task1, TIME_QUANTUM},{task2, TIME_QUANTUM},{task3, TIME_QUANTUM}
};void task1(void) {// 执行任务1
}void task2(void) {// 执行任务2
}void task3(void) {// 执行任务3
}void main(void) {int currentTask = 0;while (1) {if (tasks[currentTask].remainingTime > 0) {tasks[currentTask].taskFunc();  // 执行当前任务tasks[currentTask].remainingTime--;}currentTask = (currentTask + 1) % TASK_COUNT;  // 轮询下一个任务}
}

3. 中断驱动编程

中断驱动是一种利用单片机中断机制来处理事件的方法。通过配置中断源(如 GPIO、定时器、串口等),可以在事件发生时直接跳转到中断服务例程(ISR),从而实现快速响应。

实现思路:
  • 配置中断源,设置中断优先级。

  • 在中断服务例程中处理事件,避免在 ISR 中执行耗时操作。

示例代码:

void EXTI0_IRQHandler(void) {if (EXTI_GetITStatus(EXTI_Line0) != RESET) {// 处理按键中断事件EXTI_ClearITPendingBit(EXTI_Line0);handle_button_press();}
}void handle_button_press(void) {// 按键处理逻辑
}void main(void) {// 初始化中断NVIC_EnableIRQ(EXTI0_IRQn);while (1) {// 主循环可以执行其他任务}
}

4. 非阻塞式编程

非阻塞式编程通过轮询或定时器检测事件状态,而不是在事件未发生时阻塞程序。这种方式可以提高程序的响应速度,避免因等待某个事件而导致程序卡顿。

实现思路:
  • 使用定时器或计数器检测事件状态。

  • 在主循环中不断检查事件是否发生,并根据状态执行相应操作。

示例代码:

#include "bsp_dwt.h"  // 假设使用硬件定时器库#define TIMEOUT 100000  // 超时时间,单位为微秒void handle_button_press(void) {// 按键处理逻辑
}void handle_timeout(void) {// 超时处理逻辑
}void main(void) {uint32_t start_time = DWT_GetTimeline_us();  // 获取当前时间while (1) {if (GPIO_ReadInputDataBit(GPIOB, GPIO_Pin_0) == SET) {// 按键按下,执行任务handle_button_press();start_time = DWT_GetTimeline_us();  // 重置计时器}if (DWT_GetTimeline_us() - start_time > TIMEOUT) {// 超时处理handle_timeout();start_time = DWT_GetTimeline_us();  // 重置计时器}}
}

5. 超级循环(Super Loop)

超级循环是一种简单的多任务实现方式,通过在一个大循环中轮流执行不同的任务。每个任务函数执行一个任务的一部分,然后将控制权交给下一个任务。

实现思路:
  • 在主循环中按顺序调用各个任务函数。

  • 可以通过条件语句或计数器控制任务的执行频率。

示例代码:

void task1(void) {// 执行任务1
}void task2(void) {// 执行任务2
}void main(void) {while (1) {task1();task2();}
}

总结

在单片机裸机编程中,状态机、时间片轮询、中断驱动、非阻塞式编程和超级循环都是常见的编程模式。它们各有优缺点,适用于不同的场景:

  • 状态机:适用于复杂逻辑和事件驱动的系统,能够有效管理任务切换和逻辑复杂性。

  • 时间片轮询:适合多任务并发但对实时性要求不高的场景。

  • 中断驱动:适合对实时性要求较高的系统。

  • 非阻塞式编程:适合需要快速响应多个事件的系统。

  • 超级循环:适合简单的多任务场景。

相关文章:

单片机裸机编程:状态机与其他高效编程框架

在单片机裸机编程中,状态机是一种非常强大的工具,能够有效管理复杂的逻辑和任务切换。除了状态机,还有其他几种编程模式可以在不使用 RTOS 的情况下实现高效的程序设计。以下是一些常见的方法: 1. 状态机编程 状态机通过定义系统…...

图表控件Aspose.Diagram入门教程:使用 Python 将 VSDX 转换为 PDF

将VSDX转换为PDF可让用户轻松共享图表。PDF 文件保留原始文档的布局和设计。它们广泛用于演示文稿、报告和文档。在这篇博文中,我们将探讨如何在 Python 中将 VSDX 转换为 PDF。 本文涵盖以下主题: Python VSDX 到 PDF 转换器库使用 Python 将 VSDX 转…...

DPVS-1:编译安装DPVS (ubuntu22.04)

操作系统 rootubuntu22:~# lsb_release -a No LSB modules are available. Distributor ID: Ubuntu Description: Ubuntu 22.04.3 LTS Release: 22.04 Codename: jammy rootubuntu22:~# 前置软件准备 apt install git apt install meson apt install gcc ap…...

即将发布书籍 - Yocto项目实战教程:高效定制嵌入式Linux系统

以下这本书《Yocto项目实战教程:高效定制嵌入式Linux系统》即将发布,现在请哪位大佬出山写一个序或者推荐,有兴趣的大佬,请联系我! Git仓库地址: https://github.com/jerrysundev/Yocto-Project-Book.git …...

Git 常用指令及其说明

配置相关 # 配置全局用户名 git config --global user.name "YourUsername"# 配置全局邮箱 git config --global user.email "your.emailexample.com"说明:这两条命令用于设置 Git 全局的用户名和邮箱,在提交代码时,这些…...

nginx代理后502

直接访问 ​https://dashscope.aliyuncs.com/compatible-mode/v1/chat/completions正常 使用nginx代理后访问出现502 server {listen 9999;server_name 172.21.3.78;location ^~ /compatible-mode {proxy_pass https://dashscope.aliyuncs.com;}location / {proxy_pass…...

大模型WebUI:Gradio全解12——LangChain原理及其agent构建Gradio(1)

大模型WebUI:Gradio全解12——LangChain原理及其agent构建Gradio(1) 前言本篇摘要12. LangChain原理及其agent构建Gradio12.1 LangChain概念及优势分析12.1.1 概念12.1.2 标准化组件接口1. 示例:聊天模型2. 示例:检索器12.1.3 编排组件12.1.4 便于部署12.1.5 可观测性和评…...

【Unity】鱼群效果模拟

鱼群效果模拟 文章目录 鱼群效果模拟Boid算法实现方式version1_CPUversion2_GPUversion3_Multilaterationversion4_Bitonic_Sorting (GPU友好)version5_Skinning (TODO) 细节项优化项参考链接 Boid算法 Boid算法是一种模拟群体行…...

PHP入门基础学习五(函数1)

函数 一、概念 1、什么是函数? 函数:封装一段用于完成特定功能的代码 当使用一个函数时,只需关心函数的参数和返回值,就可以完成一个特定的功能 2、php中的函数 PHP 的真正威力源自于它的函数,PHP 中提供了超过 1000 个内建的函数。 php函数分为: 系统内部函数和自…...

微信小程序 - 页面跳转(wx.navigateTo、wx.redirectTo、wx.switchTab、wx.reLaunch)

API 跳转 1、wx.navigateTo (1)基本介绍 功能:保留当前页面,跳转到应用内的某个页面,使用该方法跳转后可以通过返回按钮返回到原页面 使用场景:适用于需要保留当前页面状态,后续还需返回的情…...

Typora的Github主题美化

[!note] Typora的Github主题进行一些自己喜欢的修改,主要包括:字体、代码块、表格样式 美化前: 美化后: 一、字体更换 之前便看上了「中文网字计划」的「朱雀仿宋」字体,于是一直想更换字体,奈何自己拖延症…...

2.3 变量

版权声明:本文为博主原创文章,转载请在显著位置标明本文出处以及作者网名,未经作者允许不得用于商业目的。 变量是用来存放某个值的数据,它可以表示一个数字、一个字符串、一个结构、一个类等。变量包含名称、类型和值。在代码中…...

Docker:Docker从入门到精通(一)- Docker简介

一、前言 通过本专栏的学习,我们将了解   1. 掌握Docker基础知识,能够理解Docker镜像与容器的概念   2. 完成Docker安装与启动   3. 掌握Docker镜像与容器相关命令   4. 掌握Tomcat Nginx 等软件的常用应用的安装   5. 掌握docker迁移与备份相…...

【复习】Redis

数据结构 Redis常见的数据结构 String&#xff1a;缓存对象Hash&#xff1a;缓存对象、购物车List&#xff1a;消息队列Set&#xff1a;点赞、共同关注ZSet&#xff1a;排序 Zset底层&#xff1f; Zset底层的数据结构是由压缩链表或跳表实现的 如果有序集合的元素 < 12…...

在Spring Boot+Vue前后端分离的项目中使用JWT实现基本的权限校验

说明 在 Spring Boot + Vue 前后端分离的项目中,如果不使用第三方服务(如 Spring Security、Shiro 等),可以通过自定义实现基本的权限校验。 使用JWT实现步骤 以下是实现步骤: 1. 设计权限模型 通常权限模型包括: 用户(User):系统的使用者。角色(Role):用户的权…...

蓝桥杯单片机组第十二届省赛第二批次

前言 第十二届省赛涉及知识点&#xff1a;NE555频率数据读取&#xff0c;NE555频率转换周期&#xff0c;PCF8591同时测量光敏电阻和电位器的电压、按键长短按判断。 本试题涉及模块较少&#xff0c;题目不难&#xff0c;基本上准备充分的都能完整的实现每一个功能&#xff0c;并…...

伪404兼容huawei生效显示404

根据上述思考&#xff0c;以下是详细的中文分步说明&#xff1a; --- **步骤 1&#xff1a;获取目标设备的User-Agent信息** 首先&#xff0c;我们需要收集目标设备的User-Agent字符串&#xff0c;包括&#xff1a; 1. **iPhone设备的User-Agent**&#xff1a; Mozi…...

UIAutomation开发常用方法的参考文档

简介 由于UIAutomation的官方文档只有一个github中的readme文件,只是简单的使用示例,具体使用还需要在代码中查找,非常不方便。经过我多年使用UIAutomation开发的经验和整理,把常用的功能梳理成本文档,作为我的开发参考使用,这样就不用每次都翻代码了,同时也可以使用AI…...

数据库面试题(基础常考!!!)

在数据库领域&#xff0c;无论是日常开发还是面试场景&#xff0c;都有一些高频且重要的问题需要我们深入理解和掌握。本文将对这些常见面试题进行详细阐述&#xff0c;帮助大家更好地应对面试和实际工作中的挑战。 面试题一&#xff1a;三范式详解 什么是三范式 三范式是关…...

ASP.NET Core Clean Architecture

文章目录 项目地址一、项目主体1. CQRS1.1 Repository数据库接口1.2 GetEventDetail 完整的Query流程1.3 创建CreateEventCommand并使用validation 2. EFcore层2.1 BaseRepository2.2 CategoryRepository2.3 OrderRepository 3. Email/Excel导出3.1 Email1. IEmail接口层2. Ema…...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练

前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1)&#xff1a;从基础到实战的深度解析-CSDN博客&#xff0c;但实际面试中&#xff0c;企业更关注候选人对复杂场景的应对能力&#xff08;如多设备并发扫描、低功耗与高发现率的平衡&#xff09;和前沿技术的…...

Golang dig框架与GraphQL的完美结合

将 Go 的 Dig 依赖注入框架与 GraphQL 结合使用&#xff0c;可以显著提升应用程序的可维护性、可测试性以及灵活性。 Dig 是一个强大的依赖注入容器&#xff0c;能够帮助开发者更好地管理复杂的依赖关系&#xff0c;而 GraphQL 则是一种用于 API 的查询语言&#xff0c;能够提…...

使用Matplotlib创建炫酷的3D散点图:数据可视化的新维度

文章目录 基础实现代码代码解析进阶技巧1. 自定义点的大小和颜色2. 添加图例和样式美化3. 真实数据应用示例实用技巧与注意事项完整示例(带样式)应用场景在数据科学和可视化领域,三维图形能为我们提供更丰富的数据洞察。本文将手把手教你如何使用Python的Matplotlib库创建引…...

SiFli 52把Imagie图片,Font字体资源放在指定位置,编译成指定img.bin和font.bin的问题

分区配置 (ptab.json) img 属性介绍&#xff1a; img 属性指定分区存放的 image 名称&#xff0c;指定的 image 名称必须是当前工程生成的 binary 。 如果 binary 有多个文件&#xff0c;则以 proj_name:binary_name 格式指定文件名&#xff0c; proj_name 为工程 名&…...

【Linux】Linux 系统默认的目录及作用说明

博主介绍&#xff1a;✌全网粉丝23W&#xff0c;CSDN博客专家、Java领域优质创作者&#xff0c;掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域✌ 技术范围&#xff1a;SpringBoot、SpringCloud、Vue、SSM、HTML、Nodejs、Python、MySQL、PostgreSQL、大数据、物…...

基于鸿蒙(HarmonyOS5)的打车小程序

1. 开发环境准备 安装DevEco Studio (鸿蒙官方IDE)配置HarmonyOS SDK申请开发者账号和必要的API密钥 2. 项目结构设计 ├── entry │ ├── src │ │ ├── main │ │ │ ├── ets │ │ │ │ ├── pages │ │ │ │ │ ├── H…...

Kubernetes 节点自动伸缩(Cluster Autoscaler)原理与实践

在 Kubernetes 集群中&#xff0c;如何在保障应用高可用的同时有效地管理资源&#xff0c;一直是运维人员和开发者关注的重点。随着微服务架构的普及&#xff0c;集群内各个服务的负载波动日趋明显&#xff0c;传统的手动扩缩容方式已无法满足实时性和弹性需求。 Cluster Auto…...

数据库正常,但后端收不到数据原因及解决

从代码和日志来看&#xff0c;后端SQL查询确实返回了数据&#xff0c;但最终user对象却为null。这表明查询结果没有正确映射到User对象上。 在前后端分离&#xff0c;并且ai辅助开发的时候&#xff0c;很容易出现前后端变量名不一致情况&#xff0c;还不报错&#xff0c;只是单…...

k8s从入门到放弃之Pod的容器探针检测

k8s从入门到放弃之Pod的容器探针检测 在Kubernetes&#xff08;简称K8s&#xff09;中&#xff0c;容器探测是指kubelet对容器执行定期诊断的过程&#xff0c;以确保容器中的应用程序处于预期的状态。这些探测是保障应用健康和高可用性的重要机制。Kubernetes提供了两种种类型…...

二维数组 行列混淆区分 js

二维数组定义 行 row&#xff1a;是“横着的一整行” 列 column&#xff1a;是“竖着的一整列” 在 JavaScript 里访问二维数组 grid[i][j] 表示 第i行第j列的元素 let grid [[1, 2, 3], // 第0行[4, 5, 6], // 第1行[7, 8, 9] // 第2行 ];// grid[i][j] 表示 第i行第j列的…...