0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

LEDs状态灯任务(线程)设计 (基于RTOS)

黄工的嵌入式技术圈 来源:网站整理 2020-03-12 11:30 次阅读
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

我们学习MCU开发,大部分都是面向过程的开发,但实际项目一般要求我们有面向对象(模块化)的方式来开发。

刚学习C语言开发的朋友,应该常常听说面向对象,但实际对于面向对象开发可能还是不太了解。

为了初学者进一步理解,本文结合实际项目(LEDs状态灯)给大家带来比较基础的模块化设计。

Ⅰ关于C语言的模块化

对于MCU的开发,大部分人都还是习惯性用的C语言,原因之一在于C语言具有高效的特点。

可以了解一下,许多操作系统的内核使用的编程语言,其实都用到了C语言,这就是C语言的优点,也是C语言这么多年不衰败的原因。

说回来,对于MCU的开发,除了C语言,当然还可以其它语言,像C++有许多人就用上了。

C++语言本身就是面向对象的开发语言,定义一个类,可以包含许多成员。站在C语言的角度,可以理解成定义一个结构体,里面包含许多数据类型。如下面要说的LEDs数据结构体:

typedef struct { uint8_t Mode; //模式(常灭 常亮 闪烁) uint8_t Status; //当前状态(灭 亮) uint16_t OffTimes; //灭时间(xLED_COUNT_PERIOD毫秒) uint16_t OnTimes; //亮时间 uint16_t Counter; //计数(计时) void (*OffFun)(void); //灭函数接口 void (*OnFun)(void); //亮函数接口 }LED_TypeDef;

可以看到,结构体里面包含整型变量,函数指针。

补充,指针函数与函数指针的区别:

1、指针函数:本质是一个函数,函数返回类型是某一类型的指针。

格式: 类型标识符 *函数名(参数表)

如:int *f(x,y);

2、函数指针:本质是一个指针,指向函数的指针变量。

格式:类型说明符 (*函数名)(参数)

如:int (*f) (int x);

Ⅱ为什么要模块化设计

假如一个系统中做的事情非常多,比如:采集两个增量式编码器、两个绝对值编码器、控制4个电机、控制多个LED状态灯、通信收发数据,采集温度、湿度、超声波雷达等···许多模块,那么问题来了,这么多模块,你的软件该如何设计?

答案就是需要模块化设计。

模块化设计,包含底层驱动,中间接口函数,应用程序等。对于MCU级别的开发,为了规范,建议大家从底层设计到应用层设计都按照模块化的方式来设计。

简单的来说,模块化就是源文件、数据结构、变量、函数命名等需要按照模块的方式来设计。比如LEDs状态灯:IO口的定义用LED(模块),文件名用led,变量、函数名抬头用LED,定义一个LED数据结构(模块的数据结构)等。

模块化的设计优点在于:便于源代码管理、移植、理解等等。(相信有许多自己写的代码,放一段时间之后,重新再次阅读,可能看了半天都不明白源代码的意思。)

ⅢLEDs实例讲述

为方便大家理解,拿一个简单的LEDs状态灯的实例来分析。里面使用到了RTOS简单系统延时(本文不讲述关于RTOS的知识)。文末提供例程下载地址。

1.描述

绿、黄、红三个(可以自己添加许多个)LED状态灯,可独自实现常灭、常亮、闪烁三个模式。

闪烁:灭、亮时间可设置(提供函数接口修改)。

在一个线程(任务)里面执行。

3个LED不同亮灭时间效果:

2.数据结构

typedef struct { uint8_t Mode; //模式(常灭 常亮 闪烁) uint8_t Status; //当前状态(灭 亮) uint16_t OffTimes; //灭时间(xLED_COUNT_PERIOD毫秒) uint16_t OnTimes; //亮时间 uint16_t Counter; //计数(计时) void (*OffFun)(void); //灭函数接口 void (*OnFun)(void); //亮函数接口 }LED_TypeDef;

为了方便理解,只使用一个数据结构(实际大的项目可能有多个包含,类似C++继承关系)。

3.底层LED函数接口

void LEDGreen_Off(void);

void LEDGreen_On(void);

void LEDYellow_Off(void);

void LEDYellow_On(void);

void LEDRed_Off(void);

void LEDRed_On(void);

主要就是亮灭函数接口,这里提供三组LED(根据需求可添加)。

4.定义局部变量

static LED_TypeDef sLEDG_Structure; //绿灯 static LED_TypeDef sLEDY_Structure; //黄灯 static LED_TypeDef sLEDR_Structure; //红灯

5.初始化变量

/************************************************函数名称 : LED_Data_Init功 能 : 数据初始化参 数 : 无返 回 值 : 无作 者 : strongerHuang*************************************************/ static void LED_Data_Init(void){ /* 绿灯 */ sLEDG_Structure.Mode = LED_MODE_FLICKER; //初始化为闪烁 sLEDG_Structure.OffTimes = 50; //灭亮时间 sLEDG_Structure.OnTimes = 50; sLEDG_Structure.Counter = 0; //计数归零 sLEDG_Structure.OffFun = LEDGreen_Off; //灭函数接口 sLEDG_Structure.OnFun = LEDGreen_On; //亮函数接口 /* 黄灯 */ sLEDY_Structure.Mode = LED_MODE_ON; //初始化为常亮 sLEDY_Structure.OffTimes = 0; //灭亮时间 sLEDY_Structure.OnTimes = 0; sLEDY_Structure.Counter = 0; //计数归零 sLEDY_Structure.OffFun = LEDYellow_Off; //灭函数接口 sLEDY_Structure.OnFun = LEDYellow_On; //亮函数接口 /* 红灯 */ sLEDR_Structure.Mode = LED_MODE_ON; //初始化为常亮 sLEDR_Structure.OffTimes = 0; //灭亮时间 sLEDR_Structure.OnTimes = 0; sLEDR_Structure.Counter = 0; //计数归零 sLEDR_Structure.OffFun = LEDRed_Off; //灭函数接口 sLEDR_Structure.OnFun = LEDRed_On; //亮函数接口 /* 对外调用接口(例子) */ LEDG_Set(LED_MODE_FLICKER, 50, 50); LEDY_Set(LED_MODE_FLICKER, 50, 10); LEDR_Set(LED_MODE_FLICKER, 20, 30);}

这里重要的就是要初始化灭亮函数接口。

6.LEDs任务(线程)

/************************************************函数名称 : LED_Task_Proc功 能 : 状态灯任务程序参 数 : pvParameters --- 可选参数返 回 值 : 无作 者 : strongerHuang*************************************************/ static void LED_Task_Proc(void *pvParameters){ static TickType_t xLastWakeTime; xLastWakeTime = xTaskGetTickCount(); for(;;) { //间隔固定计数周期(采样时间) vTaskDelayUntil(&xLastWakeTime, LED_COUNT_PERIOD); /* 浏览LEDs */ LED_Scan(&sLEDG_Structure); LED_Scan(&sLEDY_Structure); LED_Scan(&sLEDR_Structure); }}

流程图:

7.LED浏览(或者说处理)

/************************************************函数名称 : LED_Scan功 能 : 状态灯扫描(修改状态)参 数 : LED_Struct --- 状态灯数据结构返 回 值 : 无作 者 : strongerHuang*************************************************/ static void LED_Scan(LED_TypeDef *LED_Struct){ /* 1.常灭模式 */ if(LED_MODE_OFF == LED_Struct->Mode) { LED_Struct->Status = LED_STATUS_OFF; //状态置为"灭" LED_Struct->OffFun(); //灭灯 } /* 2.常亮模式 */ else if(LED_MODE_ON == LED_Struct->Mode) { LED_Struct->Status = LED_STATUS_ON; //状态置为"亮" LED_Struct->OnFun(); //亮灯 } /* 3.闪烁模式 */ else if(LED_MODE_FLICKER == LED_Struct->Mode) { /* 在灭的状态 */ if(LED_STATUS_OFF == LED_Struct->Status) { LED_Struct->Counter++; if(LED_Struct->Counter >= LED_Struct->OffTimes) { LED_Struct->Counter = 0; LED_Struct->OnFun(); //亮灯 LED_Struct->Status = LED_STATUS_ON; //状态置为"亮" } } /* 在亮的状态 */ else if(LED_STATUS_ON == LED_Struct->Status) { LED_Struct->Counter++; if(LED_Struct->Counter >= LED_Struct->OnTimes) { LED_Struct->Counter = 0; LED_Struct->OffFun(); //灭灯 LED_Struct->Status = LED_STATUS_OFF; //状态置为"灭" } } else { LED_Struct->Status = LED_STATUS_OFF; //状态置为"灭" } } /* 4.未知模式 */ else { LED_Struct->Status = LED_STATUS_OFF; //状态置为"灭" LED_Struct->OffFun(); //灭灯 }}

源代码工程下载地址:

链接:https://pan.baidu.com/s/1cNtwJDdCOfyYwsvKCclFyw

密码:kk74

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • mcu
    mcu
    +关注

    关注

    147

    文章

    18618

    浏览量

    387340
  • LEDs
    +关注

    关注

    1

    文章

    42

    浏览量

    26156
  • RTOS
    +关注

    关注

    25

    文章

    862

    浏览量

    122616
  • 线程
    +关注

    关注

    0

    文章

    508

    浏览量

    20761
收藏 人收藏
加入交流群
微信小助手二维码

扫码添加小助手

加入工程师交流群

    评论

    相关推荐
    热点推荐

    线程的系统

    线程系统的事件响应也是在中断中完成的,但事件的处理是在线程中完成的。在多线程系统中,线程跟中断一样,也具有优先级,优先级高的线程会被优先执
    发表于 12-08 07:55

    RTOS Crash 问题全维度分析与解决指南

    ()/vTaskGetRunTimeStats(),打印任务状态、栈剩余空间、CPU使用率,快速识别死锁/资源耗尽; 错误码捕获 :对所有RTOS API返回值做检查(如xTaskCreate返回pdFAIL),避免忽略资源创建
    发表于 12-08 03:56

    FreeRTOS 空闲任务

    几乎所有的小型 RTOS 中都会有一个空闲任务,空闲任务属于系统任务,是必须要执行的,用户程序不能将其关闭。不光小型系统中有空闲任务,大型的
    发表于 12-04 07:35

    Linux多线程对比单线程的优势

    :「资源利用率」:通过多线程,可以更有效地利用CPU资源,特别是多核CPU。「并行处理」:线程允许同时执行多个任务,提高程序的执行效率。「简化设计」:使用线程可以简化程序设计,因为
    发表于 12-01 06:11

    RTOS 必学概念:任务、信号量、队列一次搞懂

    如果你刚接触RTOS(实时操作系统),很可能会有这样的困惑:“RTOS和裸机程序到底有什么区别?”“任务线程吗?为什么要分任务?”“信号量
    的头像 发表于 11-17 10:53 197次阅读
    <b class='flag-5'>RTOS</b> 必学概念:<b class='flag-5'>任务</b>、信号量、队列一次搞懂

    【HZ-T536开发板免费体验】—— linux创建线程

    线程与进程 一个进程指的是一个正在执行的应用程序,而线程的功能是执行应用程序中的某个具体任务线程具有传统线程的特征,一个进程包括多个
    发表于 09-01 21:31

    Task任务:LuatOS实现“任务级并发”的核心引擎

    Task任务通过其强大的并发处理能力,使LuatOS能够在单线程环境中模拟多线程执行,通过协程的挂起与恢复机制,实现任务级的并行操作,显著提升系统效能。 sys核心库是LuatOS运行
    的头像 发表于 08-28 13:49 340次阅读
    Task<b class='flag-5'>任务</b>:LuatOS实现“<b class='flag-5'>任务</b>级并发”的核心引擎

    同步任务开发指导

    同步任务是指在多个线程之间协调执行的任务,其目的是确保多个任务按照一定的顺序和规则执行,例如使用锁来防止数据竞争。 同步任务的实现需要考虑多
    发表于 06-19 07:57

    CPU密集型任务开发指导

    CPU密集型任务是指需要占用系统资源处理大量计算能力的任务,需要长时间运行,这段时间会阻塞线程其它事件的处理,不适宜放在主线程进行。例如图像处理、视频编码、数据分析等。 基于多
    发表于 06-19 06:05

    工控一体机多线程任务调度优化:聚徽分享破解工业复杂流程高效协同密码

    在当今工业 4.0 的浪潮下,工业生产正朝着高度自动化、智能化的方向大步迈进。生产流程日益复杂,众多任务需要同时、高效地协同执行,这对工业控制系统的核心 —— 工控一体机提出了前所未有的挑战。多线程
    的头像 发表于 05-28 14:06 487次阅读

    RTOS如何在FX3中工作?

    大家好, 我正在使用 FX3 进行一个项目。 我想知道 RTOS 调度是如何工作的。 我知道调用“CyU3PKernelEntry();”后 RTOS 就会开始工作。 如果我只注册一个应用程序线程。 我的
    发表于 05-06 13:20

    详解RTOS中的Hook函数

    Hook函数是RTOS中的一个关键特性,通过该函数,用户可以增强对任务管理的控制,定义系统行为。
    的头像 发表于 03-24 16:14 841次阅读

    RTOS中的本地存储指针使用

    本地存储指针是RTOS中的一个重要特性,增强了任务管理和数据处理能力。在RTOS上下文中,本地存储是指存储在本地的特定任务或对象的数据。通常与任务
    的头像 发表于 02-28 16:33 1172次阅读
    <b class='flag-5'>RTOS</b>中的本地存储指针使用

    EE-303:将VisualDSP线程安全库与第三方RTOS配合使用

    电子发烧友网站提供《EE-303:将VisualDSP线程安全库与第三方RTOS配合使用.pdf》资料免费下载
    发表于 01-07 14:09 0次下载
    EE-303:将VisualDSP<b class='flag-5'>线程</b>安全库与第三方<b class='flag-5'>RTOS</b>配合使用

    使用任务通知提高RTOS应用的效率

    在实时嵌入式系统中,性能和资源效率是决定设计成败的关键因素。传统的实时操作系统(RTOS)提供了如队列、信号量和事件组机制,实现任务之间的同步和通信。FreeRTOS/SAFERTOS还提供一种方法可以使这些过程更快、更轻量化,即任务
    的头像 发表于 12-27 14:54 1073次阅读