IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 嵌入式 -> 适合单片机使用的一个短小精悍的时间片轮询系统 -> 正文阅读

[嵌入式]适合单片机使用的一个短小精悍的时间片轮询系统

????????随着单片机外设的日益丰富,以及RAM,ROM的增大。RTOS实时操作系统被越来越广泛的使用。实时操作系统对任务的实时性高效处理是毋容置疑的!

但是有更多的情况我们仅仅需要一个短小精悍的轮询系统,比如实时要求不高的任务,受限内存的51系列单片机,任务较少的项目,入门不不久新朋友。相对于实时操作系统,裸机开发更简单,更快捷,代码更精简,代码执行效率更高!在逻辑不是特别复杂,要求不是特别高的情况下,裸机开发反而更适合项目需求!废话不多说先上连接点此免费下载

????????源代码在STC官方库的基础上修改,特此声明。如果仅作时间片轮询可直接使用STC官方库,我上传的源码也是可用的。但不一定刚好完美适合你,有动手能力的可以自行下载优化。

? ? ? ? 下边是代码讲解:

????????首先在Task.h文件中定义一个任务结构:

第一个元素是任务就绪标志位,可以手动置位,也可以定时器置位。第二个元素是挂起标志位,可以手动置位。

????????因为在执行回调函数中这两个标志位是逻辑与关系,所以只有两个都置1任务才会执行,一般使用时Run由定时器控制,Suspend由人工控制。不需要挂起可以直接删掉或者全部默认1。优先级后边讲解,不同的TIMCount可保证不同的任务分时启动,错开周期。TRITime周期重装值关系到任务执行的间隔时间!*TaskHook指向任务函数的指针,必须是无参数,无返回值类型。而且一定不能有死循环或者阻塞式延时函数。

typedef struct 
{
	uint8_t 	Run;               		//任务状态:Run/Stop
	uint8_t		Suspend;				//挂起任务
	uint8_t  	Priority;				//任务优先级
	uint16_t 	TIMCount;         		//定时计数器
	uint16_t 	TRITime;          		//重载计数器
	void (*TaskHook) (void); 			//任务函数
} TASK_COMPONENTS;  

使用头文件定义的结构体定义一个任务队列数组,把需要执行的任务填到数组中,越靠前执行的基础优先级越高。最后通过sizeof计算数组长度。

 TASK_COMPONENTS Task_Comps[]=
{
//	 状态  挂起      优先级      计数     周期  			函数
//	{Run, Suspend, Priority, TIMCount,TRITime,void (*TaskHook) (void);},						        
    {0, 1, 5, 25, 25, run_link},				    	/* 运行指示灯闪烁:250ms*/
	{0, 1, 1, 5, 5, Duty_cycle_updata},				/* 占空比更新,浓度计算更新:50ms */
	{0, 1, 1, 5, 5, ADC_data_handling},				/* ADC相关的数据处理 */
	{0, 1, 1, 1, 10, Can0_tx1},						/* 数据通信CAN第1帧  100ms */
	{0, 1, 1, 2, 10, Can0_tx2},						/* 数据通信CAN第2帧   100ms */
	{0, 1, 1, 3, 10, Can0_tx3},						/* 数据通信CAN第3帧   100ms */
	{0, 1, 1, 4, 10, Can0_tx4},						/* 数据通信CAN第4帧   100ms */
	{0, 1, 1, 5, 50, Can0_tx5},						/* 数据通信CAN第5帧   500ms */
	{0, 1, 1, 6, 50, Can0_tx6},						/* 数据通信CAN第6帧   500ms */
	{0, 1, 1, 7, 50, Can0_tx7},						/* 数据通信CAN第7帧   500ms */
	{0, 1, 1, 8, 50, Can0_tx8},						/* 数据通信CAN第8帧   500ms */
	{0, 1, 0, 10, 10, power_check}					/* 电源检测: 100ms */
};

uint8_t Tasks_Max = sizeof(Task_Comps)/sizeof(Task_Comps[0]);

?接下来我们要在定时器中断中做任务的就绪标记,建议使用1MS定时器。每次进入中断,我们对每个任务的计数器自减,一旦计数器归零,我们标记任务就绪,同时重装计数器为下次做准备。这里的重装值其实可以在任务的执行过程更改。我们可以随时改变任务的执行周期,比如我们可以在高负荷任务执行时,修改部分不重要的任务执行周期!以达到效率最大化目的!

//========================================================================
// 函数: Task_Handler_Callback
// 描述: 任务标记回调函数.
// 参数: None.
// 返回: None.
// 版本: V1.1, 2022-04-15
//========================================================================
void Task_Marks_Handler_Callback(void)
{
	uint8_t i;
	for(i=0; i<Tasks_Max; i++)
	{
		if(Task_Comps[i].TIMCount)    /* If the time is not 0 */
		{
			Task_Comps[i].TIMCount--;  /* Time counter decrement */
			if(Task_Comps[i].TIMCount == 0)  /* If time arrives */
			{
				/*Resume the timer value and try again */
				Task_Comps[i].TIMCount = Task_Comps[i].TRITime;  
				Task_Comps[i].Run = 1;    /* The task can be run */
			}
		}
	}
}

当任务标记完成后,我们就可以在主函数中轮询对任务执行了,具体如下:

首先从最高优先级的队列最靠前的查询,如果就绪标志和挂起标志都是1,那么就执行一次任务,同时返回任务标号,这个任务标号可以在主循环打印出来或记录,作为日志使用!

//========================================================================
// 函数: Task_Pro_Handler_Callback
// 描述: 任务处理回调函数.
// 参数: None.
// 返回: 任务队列编号(数组下标),可根据返回值检测程序是否按预计流程执行.
// 版本: V1.1, 2022-04-15
//========================================================================
uint8_t Task_Pro_Handler_Callback(void)
{
	uint8_t i,j;
	uint8_t k;

	for (i = 0; i < Priority_Max; ++i)		//从最高优先级查询
	{
		for(j=0; j<Tasks_Max; j++)			//从任务队列最前查询
		{
			if (Task_Comps[j].Priority==i)	//当前任务符合当前优先级
			{
				if((Task_Comps[i].Run) && (Task_Comps[i].Suspend))/* 如果任务就绪并且未挂起 */
				{
					Task_Comps[i].Run = 0;    /* 重置就绪标志*/
					Task_Comps[i].TaskHook();  /* 运行任务 */

					return i;				//返回执行的任务编号,结束本次查询执行
				}
			}
		}
	  }
	return 0xff;	//未执行任何任务,此处可不要。
}

到此配合我们上次讲的非阻塞串口收发,即可成为一个非常短小精悍的逻辑轮询式开发框架。51单片机非阻塞串口中断收发数据

  嵌入式 最新文章
基于高精度单片机开发红外测温仪方案
89C51单片机与DAC0832
基于51单片机宠物自动投料喂食器控制系统仿
《痞子衡嵌入式半月刊》 第 68 期
多思计组实验实验七 简单模型机实验
CSC7720
启明智显分享| ESP32学习笔记参考--PWM(脉冲
STM32初探
STM32 总结
【STM32】CubeMX例程四---定时器中断(附工
上一篇文章      下一篇文章      查看所有文章
加:2022-05-18 17:49:03  更:2022-05-18 17:50:37 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 12:04:12-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码