单片机裸机程序——程序架构

news2025/1/15 6:55:18

目 录

  • 程序架构等同于思想体系
  • 一、前后台顺序法
  • 二、时间片轮询法


程序架构等同于思想体系

建一栋楼房,地基要先设计好,而不是马上砌砖,地基和布局都合理,房子就住得舒服,也不会闹心。

写一段程序也一样,程序构架要想好,而不是一边写一边调整构架,想到这个功能就先写这个功能,多个功能放在一起后无法协调,又要改构架,效率非常低。

把构架用图形的形式设计好,再像添砖加瓦一样去编码,实现功能。

一、前后台顺序法

公司都有负责接待来宾的前台,当有人来访或有电话来,前台就会通知其他负责人,正在有序工作的负责人必须马上停止手中工作,接待或接听结束后再回到原来的事情上继续。有人来访或有电话比做前台,正在有序工作比做后台。

前后台顺序执行是入门开发者常用的程序架构,逻辑简单,复杂度低,代码量少,最直观了,从上往下执行,全部任务都在一个循环里执行,不考虑每个函数执行所需要的时间。但遇到延时就要等待,因此会造成其他函数间隔执行时间的不同,尽管能够通过定时器中断的方式,但前提是中断服务函数执行的时间必须短。

顺序执行就是在你前面的执行完了,才到你执行,好比上体育课列队报数一样,必须按顺序来,你前面相邻的同学报完数,你才能报数。

还有就是有中断发生,就优先执行中断要执行的事情,好比体育课,点到谁的名字就优先得去跑两圈。

未加入中断的轮询系统

int main(void) 
{ 
    uint8 keyValue;
 
    InitSys();                  // 系统初始化
    while (1)
    {
        TaskDisplayClock();//显示系统时钟
        keyValue = TaskKeySan();//扫描键盘
        switch (keyValue)
       {
            case x: TaskDispStatus(); break;
            ...
            default: break;
        }
    }
}

在while(1)死循环中从第一行代码往下执行到最后一行,然后返回到第一行,如此反复执行。

在轮询系统基础上加入中断处理

int flag1 = 0,flag2= 0,flag3 = 0;
int main(){
  hardwareInit();//硬件初始化
  while(1){
    if(flag1){
      doSomething1();//处理事件1
      flag1=0;//清除标志位
    }
    if(flag2){
      doSomething2();//处理事件2
      flag2=0;//清除标志位
    }
    if(flag3){
      doSomething3();//处理事件3
      flag3=0;//清除标志位
    }
  }
}
//中断处理程序1
void ISR1(void ){
  flag1=1//置位标志位//中断处理程序2
void ISR2(void ){
  flag2=1//置位标志位//中断处理程序3
void ISR3(void ){
  flag3=1//置位标志位

外部事件的响应(改变一些标志位,确保所用时间非常短)在中断里面完成,事件的处理还是要回到轮询系统中完成。

1、如果中断服务程序执行时间太长,可能导致中断嵌套非常深,容易导致栈溢出,也容易出现异常,比如死机。
2、中断只处理紧急事务!紧急事务要用中断处理!比如串口数据的接收和发送,速度太快,放在循环里处理太慢了。然而对于数据的处理,就不用着急了,可以在中断里接收完成后设置标志位,然后在主循环里根据标志位完成处理,之后手动开启相应中断去发送。

中断在这里称为前台,main()函数中的主循环称为后台。
在这里插入图片描述

二、时间片轮询法

敢言:时间片轮询法是万能的裸机程序构架

时间片法是一种多任务执行法,它通过为每个任务分配一定的执行时间片,使得所有任务都能够按照一定的时间间隔交替执行。任务执行的时间片是固定的,当一个任务的时间片用完后,系统切换到下一个任务执行。这种方式使得多个任务能够并行执行,提高了系统的资源利用率和效率。

时间片法适用于中等复杂度的嵌入式系统,可以满足对实时性要求较高的场景。它需要合理设置任务的优先级和时间片大小,以确保重要任务优先执行,并且每个任务都能在适当的时间内完成。

使用定时器设定不同的时间片,定时到了某个时间节点,就去执行对应时间片里的代码,代码执行时间不能超过时间片时间,时间片时间应该根据所执行任务的耗时来设定。比如,按键扫描,通常都需要软件防抖,顺序法是延时10ms左右再去判断,但10ms极大浪费了CPU的资源,在这段时间内CPU完全可以处理很多其他事情,时间片轮询法就能很好利用CPU资源。

世界万事万物都是基于时间这条线而存在而运作的,谁都摆脱不了与时间的关联。

时间片轮询方式也容易理解,你设定的时间到了,你就执行,没到就继续等待,各个时间片之间没有优先级。

时间片轮询法系统的实现思路(只需要一个任意的定时器)
1、在系统初始化里初始化一个定时器,假设定时中断为1ms(也可以改成10ms,根据实际需求,但需要注意,中断过于频繁效率就低,中断太长,实时性差)。
2、定义多个任务变量及标志位
3、定义任务函数
4、把任务调度添加到主循环中
5、把任务切换逻辑控制添加到中断服务中

1、初始化定时器,以51单片机为例

//定时器初始化,定时1ms中断一次
void Timer1_Init(){
	AUXR &= 0xBF;		//定时器时钟12T模式
	TMOD &= 0x0F;		//设置定时器模式
	TMOD |= 0x10;		//设置定时器模式
	TL1 = 0x30;		//设置定时初值
	TH1 = 0xF8;		//设置定时初值
	TF1 = 0;		//清除TF1标志
	TR1 = 1;		//定时器1开始计时
	ET1 = 1;       //使能定时器中断
	EA = 1;
}
//定时器中断服务函数
void TM1_Isr() interrupt 3
{	
	TL1 = 0x30;		//设置定时初值
	TH1 = 0xF8;		//设置定时初值
}

2、定义任务相关

//任务结构
typedef struct _TASK_COMPONENTS
{
	unsigned int Run; //程序运行标记: 0-不运行,1-运行
	unsigned int Timer; //计时器
	unsigned int ItvTime;  //任务运行时间间隔
	void (*TaskHook)(void); //要运行的任务函数
}TASK_COMPONENTS;//任务定义,TASK_COMPONENTS是_TASK_COMPONENTS的别名

static TASK_COMPONENTS TaskComps[]=	 //时间节拍为1ms,由定时器产生
{
	{0,10,10,TaskRfCheck},	//处理RF数据
	{0,20,20,TaskUart2Check},  //每次只处理串口2的一个字节数据
	{0,8,8,TaskRGBdata},//RGB数据处理	 刷新频率不能低于30HZ
	{0,30,30,TaskPWMout}, //刷新PWM数据
	{0,1000,1000,TaskTimers}, //计时器
};
//任务清单
typedef enum _TASK_LIST
{
	TASK_RF_CHECK,
	TASK_UART2_CHECK,
	TASK_RGB_DATA,
	TASK_UART1_PRINTF,
	TASK_TIMERS,
	TASKS_MAX,	 //总共可分配的定时任务数目
}TASK_LIST;

3、定义任务函数


void TaskRfCheck() //识别遥控器按键
{
//在此处添加代码
}
void TaskUart2Check() //解析串口2数据
{
//在此处添加代码
}
void TaskRGBdata()	//处理RGBW数据
{	
//在此处添加代码
}
void TaskPWMout()	  //刷新PWM数据
{  	
//在此处添加代码
}
void TaskTimers()  //灯带独立计时器
{
//在此处添加代码
}

4、定义任务调度函数并添加到主循环中

void TaskProcess()
{
   static unsigned char i=0;  //必须是static类型
   for(i=0;i<TASKS_MAX;i++)	//逐个任务时间处理
   {
	  if(TaskComps[i].Run)	//任务可以运行
	  {
	  	 TaskComps[i].TaskHook();  //调用任务函数
		 TaskComps[i].Run = 0; //标志清0
	  }
   }		 	
}
void SysInit_all()
{
	Timer1_Init(); //定时器1初始化,用于多个任务调度
	WDT_CONTR=0x35;//看门狗设置,超时2秒则复位
}
//主函数
int main()
{
	SysInit_all();//系统初始化
	while(1)
	{
	 	TaskProcess(); //任务处理
		WDT_CONTR |= 0x10;   //喂看门狗,防止系统复位	 		
	}
	return 1;
}

5、把任务切换逻辑控制添加到中断服务中

void TaskRemarks(void)
{
   static unsigned char i=0;   //必须是static类型
   for(i=0;i<TASKS_MAX;i++)	 //逐个任务处理
   {
	   if(TaskComps[i].Timer) //时间不为0
	   {
		  TaskComps[i].Timer--; //减去一个节拍
		  if(TaskComps[i].Timer<=0)//时间减完了
		  {
			 TaskComps[i].Timer = TaskComps[i].ItvTime;	 //恢复计时器值,重新下一次
			 TaskComps[i].Run = 1; //任务可以运行
		  }
	   }
   }
}
//每1ms就执行一次TaskRemarks()
void TM1_Isr() interrupt 3
{	
	TL1 = 0x30;		//设置定时初值
	TH1 = 0xF8;		//设置定时初值
	TaskRemarks();	//任务调度
}

时间片轮询法的流程图大致为:
在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

c++,python实现网络爬虫

前言&#xff1a; 社交网络中用户生成的海量数据&#xff0c;社交网络数据的多样性和复杂性 如何高效地从海量的数据中获取和处理我们需要的信息资源&#xff1f; 该微博爬虫能够从社交网络平台中地提取文本、图片和用户之间的转发关系&#xff0c;并将这些数据结构化存储到…

Python的Windows GUI自动化之Pywinauto(四)

引言&#xff1a; 我们上章节中打开了一个应用程序后&#xff0c;并打印了所有的控件信息&#xff0c;这些对于工具无法定位到的控件有很好的协助作用&#xff08;当然这个可以作为主要的查找控件的用法&#xff0c;也可以辅助使用&#xff0c;我一般是把这个作为辅助使用&…

【C++】初识C++模板与STL

C语法相关知识点可以通过点击以下链接进行学习一起加油&#xff01;命名空间缺省参数与函数重载C相关特性类和对象-上篇类和对象-中篇类和对象-下篇日期类C/C内存管理 本章将简单分享C模板与STL相关知识&#xff0c;与之相关更多知识将留到下次更详细地来分享给大家 &#x1f3…

MySQL与ES数据实时同步,双写一致

一、简介 在项目的开发与运维过程中&#xff0c;MySQL 是业务数据库的核心角色&#xff0c;以其强大的事务处理能力和数据完整性保障&#xff0c;支撑着系统的稳定运行。随着数据量的急剧增长和查询复杂度的不断提升&#xff0c;单一依赖 MySQL 进行高效的数据检索显得日益吃力…

centos安装软件

1.centos 安装 unrar 提示找不到 使用EPEL仓库&#xff1a; 首先&#xff0c;你需要安装EPEL仓库&#xff1a; yum install epel-release 然后&#xff0c;尝试再次安装unrar&#xff1a; yum install unrar 编译安装&#xff1a; 下载源代码&#xff1a;wget http://www.rarla…

对耳朵伤害最小的耳机类型是哪种?五款口碑绝佳机型安利!

​目前来说&#xff0c;开放式耳机应该算是对耳朵伤害最小的耳机了。当今耳机市场上&#xff0c;开放式耳机以其舒适的佩戴和创新的非入耳设计赢得了众多消费者的喜爱。这种耳机让你在聆听音乐的同时&#xff0c;还能清晰地感知周围环境的声音&#xff0c;便于与人交流&#xf…

clickhouse 原理详解

1、MPP数据库简介 1.1、什么是OLTP与OLAP&#xff1f; 1.1.1、OLTP(OnLine Transaction Processing ) 联机事务处理 系统&#xff0c;例如mysql。擅长事务处理&#xff0c;在数据操作中保持着很强的一致性和原子性 &#xff0c;能够很好的支持频繁的数据插入和修改 &#x…

Java使用Tesseract进行OCR图片文字识别

前言 在当前的文字识别技术应用中&#xff0c;除了采用现有的API服务之外&#xff0c;常见的解决方案包括利用Tessdata、Canvas或OCRAD等工具。以下是对几种技术的简要分析&#xff1a; 百度API的使用体验表明&#xff0c;虽然其识别率令人满意&#xff0c;但并非完美无误。此…

一个月狂撸5.8W,利用AI制作宝宝走秀视频,让宝宝“出海”捞美刀

今天给大家分享的项目是**AI宝宝走秀视频玩法&#xff0c;**在项目拆解之前&#xff0c;先看一下这个账号。这个账号是在Tiktok上的&#xff0c;也就是海外版的抖音。 基本上都是几千万的播放&#xff0c;按照海外版抖音的激励计划&#xff0c;每播放一万次&#xff0c;就能赚8…

紧跟大模型技术趋势,为更大更通用的大模型提供底层支撑!关于智能计算系统:从深度学习到大模型,全新版本,发布!

文章目录 &#x1f4cb;前言&#x1f3af; 关于智能计算系统&#x1f3af; 内容简介&#x1f3af; 作者简介&#x1f3af; 专家推荐&#x1f3af; 目录大纲&#x1f525; 参与方式 &#x1f4cb;前言 “只要你想把大模型做得更好、做得更大、做得更快、做得更省电&#xff0c;…

翻译软件 Fastrans 开发日志 #01

目录 预览前言功能技术待办 预览 Github 仓库链接&#xff1a;https://github.com/YaoqxCN/Fastrans Gitee 仓库链接&#xff1a;https://gitee.com/yaoqx/Fastrans 求求给我点个 star 叭 qaq 现在才是 v1.0.0&#xff0c;给我个 star 鼓励我继续开发下去&#xff01; 我相信…

AI如何帮助普通人实现自我成长和副业变现

前言 最近有没有发现身边的一切都变得越来越"智能"了&#xff1f;连家里的空调都学会了自己调整温度&#xff0c;害得我每天起床都觉得它比我聪明。这不禁让我想到&#xff0c;既然连空调都在进化&#xff0c;我们这些普通人是不是也该搭上AI的快车&#xff0c;来个华…

找出所有子集异或和的和 全排列2

1863.找出所有子集异或和的和 解释&#xff1a;做本题没思路的话&#xff0c;强烈建议看本专栏上一篇博文 class Solution { public:int sum 0;int path 0;int subsetXORSum(vector<int>& nums) {dfs(nums, 0);return sum;}void dfs(vector<int>& nums,…

大模型分布式训练技术(DP、DDP和FSDP)

目录 数据并行&#xff08;PyTorch DP&#xff09; 分布式数据并行&#xff08;PyTorch DDP&#xff09; DP 与 DDP 的区别 补充说明&#xff1a;DP与DDP数据传输过程 完全分片数据并行(PyTorch FSDP) 补充说明&#xff1a;ZeRO FSDP DDP 与 FSDP 的区别 DP、DDP和FSD…

ADW400环保监测模块,用于各省市环保平台对接

ADW400环保监测模块主要用于计量低压网络的三相有功电能&#xff0c;同时可选择四个回路的电流输入&#xff0c;具有RS485通讯和470MHz无线通讯功能&#xff0c;方便用户进行用电监测、集抄和管理。可灵活安装于配电箱内&#xff0c;实现对不同区域和不同负荷的分项电能计量&am…

构建智慧园区的全方位解决方案:技术、部署与挑战应对

在当今数字化转型的浪潮中,智慧园区作为城市智能化的重要组成部分,正在成为各地政府和企业关注的焦点。本文将围绕一套完整的智慧园区解决方案,详细阐述其技术架构、部署流程以及在实施过程中可能遇到的挑战及应对策略。 1. 智慧园区解决方案概述 智慧园区解决方案旨在通过先…

Qt 一个带数据压缩的加解密实现类,压缩率达到了1/3

一.加解密效果 1-1000字符串,所占大小2890(加密前) 1-1000字符串,所占大小1964(加密后) 二.加解密功能实现类 /* Copyright (c) 2011, Andre Somers All rights reserved.Redistribution and use in…

Windows上安装 nodejs,npm 和 yarn详细教程

1、下载Node.js 访问Node.js 官网下载&#xff0c;下载需要版本版本&#xff0c;具体操作如下 2、安装Node.js 具体安装可参考以下知乎教程 https://www.zhihu.com/question/591831850/answer/3423661990 3、配置淘宝镜像 npm config set registry https://registry.npmmirror…

Codeforces Round 925 (Div. 3) E. Anna and the Valentine‘s Day Gift (博弈论*1400)

根据题目条件&#xff0c;我们知道10的m次方就是1后面跟着m个0&#xff0c;也就是说这是个最小的m1位数。 那么只要是最终得来的数的尾数是m1位数就可以通过。 思考影响数位的因素&#xff1f; 在这道题里&#xff0c;安娜能够对数位进行的操作只有删去后导零&#xff0c;由此…

安卓刷机笔记

前置知识 双清— 清除data 、 cache 四清----清除 data 、 cache 、 dalvik分区 、 system分区 四清的好处&#xff1a;卡刷ZIP格式的刷机包都是使用第三方recovery刷入的&#xff0c;所以我们在刷机之前对上一个系统的数据&#xff0c;包括用户数据、系统残留都需要彻底清除&a…