单片机应用程序的框架大概有三种:

1,简单的前后台顺序执行程序.

2,时间片轮询法.

3,应用操作系统.

下面我们主要来讲解时间片轮询法:

在这里我们先介绍一下定时器的复用功能。就是使用1个定时器,可以是任意的定时器,这里不做特殊说明,下面假设有3个任务,那么我们应该做如下工作:

1. 初始化定时器,这里假设定时器的定时中断为1ms(当然你可以改成10ms,这个和操作系统一样,中断过于频繁效率就低,中断太长,实时性差)。

2. 定义一个数值:
#define TASK_NUM   (3)                       //  这里定义的任务数为3,表示有三个任务会使用此定时器定时
uint16 TaskCount[TASK_NUM] ;           //  这里为三个任务定义三个变量来存放定时值,用于存放每个任务需要的时间
uint8  TaskMark[TASK_NUM];               //  同样对应三个标志位,为0表示时间没到,为1表示定时时间到

3. 在定时器中断服务函数中添加:
/**************************************************************************************
* FunctionName : TimerInterrupt()
* Description : 定时中断服务函数
* EntryParameter : None
* ReturnValue : None
**************************************************************************************/
void TimerInterrupt(void)                   //定时器溢出之后就会进入到这里,可以是1ms,也可以是10ms
{
    uint8 i;

for (i=0; i<TASKS_NUM; i++)        //分别去查看每个任务,这里TASKS_NUM是表示任务的个数,这里是3个
    {
        if (TaskCount[i])                         //TaskCount[i],这个是代表第i个任务当前延迟的时间,如果没有到延迟时间到0,就继续延迟
        {
              TaskCount[i]--;                     //减少一个刻度
              if (TaskCount[i] == 0)           //如果TaskCount[i]为0,说明我们指定的时间到了
              {
                    TaskMark[i] = 0x01;       //TaskMark[i],表示第i个任务的延迟已经完成,可以执行了
              }
        }
   }
}

4. 在我们的应用程序中,在需要的应用  [延迟]  的地方添加如下代码:

TaskCount[0] = 20;        // 延时20ms,比如按键程序,需要我们每20ms检测一次,并不需要实时查看
TaskMark[0]  = 0x00;     // 启动此任务的定时器

到此我们只需要在任务中判断TaskMark[0] 是否为0x01即可。其他任务添加相同,至此一个定时器的复用问题就实现了。在等待一个定时的到来的同时我们可以循环判断标志位,同时也可以去执行其他函数。  [即:在一个函数运行的间隙中我们可以运行其他的函数]

那么如果我们在一个函数延时的时候去执行其他函数,充分利用CPU时间,是不是和操作系统有些类似了呢?但是操作系统的任务管理和切换是非常复杂的。下面我们就将利用此方法架构一个新的应用程序。

时间片轮询法的架构:

1.设计一个结构体:这个是定义的一个  模子, 用于扣蛋糕用的,  实际上是定义任务包括几个要素,我们把任务的要素都列出来,

这样就可以使每个任务都有规定格式的要素了

typedef struct _TASK_COMPONENTS      //任务结构体,任务的几个要素  [这个是很重要的]
{
    uint8 Run;                                               // 程序运行标记:0-不运行,1-运行  ,这里是程序会不会运行
    uint8 Timer;                                            // 计时器, 这个是用于在定时器中减减的
    uint8 ItvTime;                                          // 任务运行间隔时间, 这个是用于更新参数的
    void (*TaskHook)(void);                          // 要运行的任务函数
} TASK_COMPONENTS;                           // 任务定义

2. 任务运行标志出来,此函数就相当于中断服务函数,需要在定时器的中断服务函数中调用此函数,这里独立出来,便于移植和理解。
/**************************************************************************************
* FunctionName   : TaskRemarks()          //放到定时器中
* Description    : 任务标志处理
* EntryParameter : None
* ReturnValue    : None
**************************************************************************************/
void TaskRemarks(void)                            //要放到  timer中断中
{
    uint8 i;
    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;            // 任务可以运行, 任务的延迟已经过了, 可以继续运行了
            }
        }
   }
}

3. 任务处理:实现任务管理操作, 用于循环判断哪个任务需要运行了
/**************************************************************************************
* FunctionName   : TaskProcess()
* Description    : 任务处理
* EntryParameter : None
* ReturnValue    : None
**************************************************************************************/
void TaskProcess(void)
{
    uint8 i;
    for (i=0; i<TASKS_MAX; i++)           // 逐个任务时间处理
    {
         if (TaskComps[i].Run)                 // 这个标志不为0, 标志为0, 标志置1是在timer时间中断中, 是在时间查询那里的
        {
             TaskComps[i].TaskHook();      // 运行任务, 这个是一个函数, 在运行完这个函数后, 我们下一步操作就是清除标志位
             TaskComps[i].Run = 0;           // 标志清0
        }
    }   
}

假设我们有三个任务:时钟显示,按键扫描,和工作状态显示

1. 定义一个上面定义的那种结构体变量:
/**************************************************************************************
* Variable definition                           
**************************************************************************************/
static TASK_COMPONENTS TaskComps[] =
{   //0:程序刚开始不运行;  60:用于减减的计数;  60:函数重复周期,用于重装减减;  TaskDisplayClock:是一个函数名,用于重复运行的刷屏;
    {0, 60, 60, TaskDisplayClock},           // 显示时钟         
    {0, 20, 20, TaskKeySan},                   // 按键扫描
    {0, 30, 30, TaskDispStatus},              // 显示工作状态
     // 这里添加你的任务。。。。
};

在定义变量时,我们已经初始化了值,这些值的初始化,非常重要,跟具体的执行时间优先级等都有关系,这个需要自己掌握。

上面的结构体初始化表示,我们有三个任务:

①每1s执行一下时钟显示,因为我们的时钟最小单位是1s,所以在秒变化后才显示一次就够了。
②由于按键在按下时电平会抖动,而我们知道一般按键的抖动大概是20ms,那么我们在顺序执行的函数中一般是延迟20ms等待电平稳定后才开始采集电平,而这里我们每20ms扫描一次,效果是非常不错的,即达到了消抖的目的,也不会漏掉按键输入。
③为了能够显示按键后的其他提示和工作界面,我们这里设计每30ms显示一次,如果你觉得反应慢了,你可以让这些值小一点。

TaskDisplayClock, TaskKeySan,  TaskDispStatus 后面的名称是对应的函数名,你必须在应用程序中编写这函数来分别完成三个任务。

2. 任务列表:(也叫作任务清单)
typedef enum _TASK_LIST
{
    TAST_DISP_CLOCK,               // 显示时钟
    TAST_KEY_SAN,                     // 按键扫描
    TASK_DISP_WS,                     // 工作状态显示
     // 这里添加你的任务。。。。
     TASKS_MAX                                           // 总的可供分配的定时任务数目
} TASK_LIST;
好好看看,我们这里定义这个任务清单的目的其实就是参数TASKS_MAX的值,其他值是没有具体的意义的,只是为了清晰的表面任务的关系而已。

硬件——STM32 , 软件框架的更多相关文章

  1. stm32软件编程的框架及注意事项——rtos篇

    0.通常,嵌入式软件(这里指单片机系统)的框架千变万化,有带rtos的,也有裸机的. 0.1.写过带系统的,也写过裸机的,这里总结一下两个类型的框架,记录下自己的心得,主要是文字描述,框架图可以后期添 ...

  2. linux内核中的GPIO系统之(1):软件框架

    一.前言 作为一个工作多年的系统工程师,免不了做两件事情:培训新员工和给新员工分配任务.对于那些刚刚从学校出来的学生,一般在开始的时候总是分配一些非常简单的任务,例如GPIO driver.LED d ...

  3. 安装 SQL Server 2008 R2 的硬件和软件要求(转)

    以下各部分列出了安装和运行 SQL Server 2008 R2 的最低硬件和软件要求.有关 SharePoint 集成模式下的 Analysis Services 的要求的详细信息,请参阅硬件和软件 ...

  4. DM8168 DVRRDK软件框架研究

    转载注明:http://blog.csdn.net/guo8113/article/details/41120491 Netra(DM8168)处理器是个多核处理器,每一个核之间相互独立却又相互关联, ...

  5. 安装 SQL Server 2012 的硬件和软件要求(官方全面)

    以下各节列出了安装和运行 SQL Server 2012 的最低硬件和软件要求. 有关 SharePoint 集成模式下 Analysis Services 的要求的详细信息,请参阅硬件和软件要求(S ...

  6. 安装 SQL Server 2005 的硬件和软件要求(官方全面)

    SQL Server 2005 安装要求 本主题介绍了安装 SQL Server 205 的硬件和软件要求,以及查看安装文档的说明. 硬件和软件要求(32 位和 64 位) 访问 SQL Server ...

  7. Niagara解决设备连接应用的软件框架平台技术。

    Niagara 是Tridium公司所研发的设计用于解决设备连接应用的软件框架平台技术. Niagara是一种应用框架,或者说是软件框架,特别设计用于应对智能设备所带来的各种挑战,包括设备连接到企业级 ...

  8. WinForm酒店管理软件--框架

    WinForm酒店管理软件--框架 搞软件开发4年多了,现在自认为还是菜鸟,从一开始走上工作岗位各种技术对我都很新奇没解决一个问题都觉得很伟大到后来开始对出路的迷茫,到现在我坚信学什么技术不重要,做什 ...

  9. STM32软件复位(基于库文件V3.5)

    源:STM32软件复位(基于库文件V3.5) void SoftReset(void) { __set_FAULTMASK(); // 关闭所有中端 NVIC_SystemReset();// 复位 ...

随机推荐

  1. 使用cocos2dx 3.2和cocosstudio屏幕适配总结----相对布局

    屏幕适配的文章太多了,基本上都是理论性的东西.大家明确了机制就知道了.没有完美的适配方案,除非你们的美工愿意折腾. 常规策略: 今天研究了一下屏幕适配导致的缩放和展示不全的问题(黑边的方案直接淘汰). ...

  2. iOS Core Animation具体解释(四)AutoLayout中的动画

    原创blog.转载请注明出处 blog.csdn.net/hello_hwc 欢迎关注我的iOS SDK具体解释专栏 http://blog.csdn.net/column/details/huang ...

  3. android 图片特效处理之图片叠加

    这篇将讲到图片特效处理的图片叠加效果.跟前面一样是对像素点进行处理,可参照前面的android图像处理系列之七--图片涂鸦,水印-图片叠加和android图像处理系列之六--给图片添加边框(下)-图片 ...

  4. #学习笔记#——JavaScript 数组部分编程(三)

    3.在数组 arr 末尾添加元素 item.不要直接修改数组 arr,结果返回新的数组 主要考察数组的concat方法,代码如下: arr.concat(item); concat 方法不修改原数组. ...

  5. 调用C#版gdal库的一个注意事项

    作者:朱金灿 来源:http://blog.csdn.net/clever101 在编译完C#版gdal库(x86平台)下,写了一个C#的控制台测试程序,出现下面的错误: 解决办法是将工程的目标平台设 ...

  6. PHP 获取完整URL地址

    /** * 获取当前完整URL * @return string */ function get_url() { $sys_protocal = isset($_SERVER['SERVER_PORT ...

  7. Python datetime time 等时间 日期 之间的计算和相互转化

    from datetime import datetime, date, timedelta, timezone from time import time, ctime, localtime, st ...

  8. 【习题 8-2 UVA-1610】Party Games

    [链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 字符串排序后 显然是n/2-1和n/2这两个字符串进行比较. 设为a,b 找到第一个不相同的位置. 即0..i-1是相同的前缀,然后 ...

  9. 深入具体解释SQL中的Null

    NULL 在计算机和编程世界中表示的是未知,不确定.尽管中文翻译为 "空", 但此空(null)非彼空(empty). Null表示的是一种未知状态.未来状态,比方小明兜里有多少钱 ...

  10. 华为畅玩5 (CUN-AL00) 刷入第三方twrp Recovery 及 root

    华为畅玩5 (CUN-AL00) 刷入第三方twrp Recovery  及 root 下载地址    http://pan.baidu.com/s/1hsn6VzA 1. 在官网申请解锁码    申 ...