找回密码
 立即注册

QQ登录

只需一步,快速开始

搜索
查看: 6937|回复: 8
收起左侧

51单片机基于一个定时器实现多个软件定时器

  [复制链接]
ID:171060 发表于 2019-11-2 23:31 | 显示全部楼层 |阅读模式
本帖最后由 lw1997 于 2019-11-2 23:33 编辑

软件定时器是用程序模拟出来的定时器,可以由一个硬件定时器模拟出成千上万个软件定时器,这样程序在需要使用较多定时器的时候就不会受限于硬件资源的不足,这是软件定时器的一个优点,即数量不受限制。但由于软件定时器是通过程序实现的,其运行和维护都需要耗费一定的CPU资源,同时精度也相对硬件定时器要差一些。在RTOS等操作系统中,都带有软件定时器,原理大同小异。典型的实现方法是:通过一个硬件定时器产生固定的时钟节拍,每次硬件定时器中断到,就对一个全局的时间标记加一,每个软件定时器都保存着到期时间,程序需要定期扫描所有运行中的软件定时器,将各个到期时间与全局时钟标记做比较,以判断对应软件定时器是否到期,到期则执行相应的回调函数,并关闭该定时器
程序如下,通过一个硬件定时器产生固定的时钟节拍,用4个LED演示效果:
  1. #include "reg51.h"
  2. //#include "STC89C5xRC.h"

  3. sbit LED1=P1^0;
  4. sbit LED2=P1^1;
  5. sbit LED3=P1^2;
  6. sbit LED4=P1^3;

  7. unsigned char softTimer_8s_flag = 0;        //8s定时器flag

  8. #define NULL        0

  9. #define SOFT_TIMER_MAX                4                //定时器个数

  10. //定义定时器ID
  11. #define ID_1S_LED_TASK                0
  12. #define ID_2S_LED_TASK                1
  13. #define ID_4S_LED_TASK                2
  14. #define ID_8S_LED_TASK                3

  15. typedef void (*pFun)(void);                //callback 函数指针类型

  16. typedef enum tmrMode {
  17.         MODE_ONE_SHOT = 0,       //单次模式
  18.         MODE_PERIODIC,           //周期模式
  19. } tmrMode;

  20. typedef enum tmrState {
  21.         SOFT_TIMER_STOPPED = 0,  //停止
  22.         SOFT_TIMER_RUNNING,      //运行
  23.         SOFT_TIMER_TIMEOUT,      //超时
  24.         SOFT_TIMER_WAITING                 //等待
  25. } tmrState;

  26. typedef struct softTimer {
  27.         unsigned char state;           //状态
  28.         unsigned char mode;            //模式
  29.         unsigned int  period;          //定时周期
  30.         unsigned int  count;                   //定时计数用
  31.         pFun callback;                                   //定时器回调函数
  32. } softTimer;

  33. //定时器结构数组
  34. softTimer softTimerList[SOFT_TIMER_MAX] = {0};

  35. //设定定时器
  36. void softTimer_Creat(unsigned char id,tmrMode mode,unsigned int interval,pFun cb)
  37. {
  38.         softTimerList[id].mode = mode;
  39.         softTimerList[id].period = interval;
  40.         softTimerList[id].count = 0;
  41.         softTimerList[id].callback = cb;
  42.         softTimerList[id].state = SOFT_TIMER_STOPPED;
  43. }

  44. //打开定时器
  45. void softTimer_Start(unsigned int id)
  46. {
  47.         softTimerList[id].state = SOFT_TIMER_RUNNING;
  48. }

  49. //停止定时器
  50. void softTimer_Stop(unsigned int id)
  51. {
  52.         softTimerList[id].state = SOFT_TIMER_STOPPED;
  53. }

  54. //清除定时器状态(用于执行事件后手动清除定时器状态)
  55. void softTimer_Clr(unsigned int id)
  56. {
  57.         if(softTimerList[id].mode == MODE_ONE_SHOT){
  58.                 softTimerList[id].state = SOFT_TIMER_STOPPED;
  59.         }else{
  60.                 softTimerList[id].state = SOFT_TIMER_RUNNING;
  61.         }
  62. }

  63. //获取定时器状态
  64. unsigned char softTimer_GetState(unsigned int id)
  65. {
  66.         return softTimerList[id].state;
  67. }

  68. void softTimer_Update(void)        //更新定时器状态,在硬件定时器中1ms调用一次
  69. {
  70.         unsigned char id;
  71.         for(id = 0 ;id <= SOFT_TIMER_MAX ; id++ ){
  72.                 switch (softTimerList[id].state){
  73.                         case SOFT_TIMER_STOPPED:
  74.                                 break;
  75.                         case SOFT_TIMER_RUNNING:
  76.                                 if(softTimerList[id].count < softTimerList[id].period){
  77.                                         softTimerList[id].count ++;
  78.                                 }else{
  79.                                         softTimerList[id].count = 0;
  80.                                         softTimerList[id].state = SOFT_TIMER_TIMEOUT;
  81.                                         softTimerList[id].callback();
  82.                                 }
  83.                                 break;
  84.                         case SOFT_TIMER_TIMEOUT:
  85.                                 if(softTimerList[id].mode == MODE_ONE_SHOT) {
  86.                                   softTimerList[id].state = SOFT_TIMER_STOPPED;
  87.                                 } else {
  88.                                   softTimerList[id].count ++;
  89.                                   softTimerList[id].state = SOFT_TIMER_RUNNING;
  90.                                 }
  91.                                 break;
  92.                         default:        //state error
  93.                           break;
  94.                 }
  95.         }
  96. }

  97. void Timer0Init(void)                //1毫秒@12.000MHz
  98. {
  99.         TMOD = 0x00;                //设置定时器模式
  100.         TL0 = 0xC0;                        //设置定时初值
  101.         TH0 = 0xE0;                        //设置定时初值
  102.         EA = 1;
  103.         ET0 = 1;
  104.         TR0 = 1;               
  105. }

  106. void Interrupt() interrupt 1
  107. {
  108.         TL0 = 0xC0;                //重载定时初值
  109.         TH0 = 0xE0;                //重载定时初值
  110.         
  111.         softTimer_Update();        //1ms更新一次softTimer
  112. }

  113. void Delayms(unsigned int ms)                //@12.000MHz
  114. {
  115.         unsigned char i, j;
  116.         while(--ms) {
  117.                 i = 2;
  118.                 j = 239;
  119.                 do {
  120.                         while (--j);
  121.                 } while (--i);
  122.         }
  123. }

  124. void softTimer_1s_cb(void)
  125. {
  126.         LED1 = ~LED1;
  127. }

  128. void softTimer_2s_cb(void)
  129. {
  130.         LED2 = ~LED2;
  131. }

  132. void softTimer_4s_cb(void)
  133. {
  134.         LED3 = ~LED3;
  135. }

  136. void softTimer_8s_cb(void)
  137. {
  138.         softTimer_8s_flag = 1;
  139. }

  140. void main()        //主函数
  141. {
  142.         LED1=0;LED2=0;LED3=0;LED4=0;
  143.         
  144.         //配置相应定时器
  145.         softTimer_Creat(ID_1S_LED_TASK,MODE_PERIODIC,1000,softTimer_1s_cb);
  146.         softTimer_Creat(ID_2S_LED_TASK,MODE_PERIODIC,2000,softTimer_2s_cb);
  147.         softTimer_Creat(ID_4S_LED_TASK,MODE_ONE_SHOT,4000,softTimer_4s_cb);
  148.         softTimer_Creat(ID_8S_LED_TASK,MODE_PERIODIC,8000,softTimer_8s_cb);
  149.         
  150.         //打开相应定时器
  151.         softTimer_Start(ID_1S_LED_TASK);
  152.         softTimer_Start(ID_2S_LED_TASK);
  153.         softTimer_Start(ID_4S_LED_TASK);
  154.         softTimer_Start(ID_8S_LED_TASK);
  155.         
  156.         Timer0Init();//初始化定时器0
  157.         while(1) {
  158.                
  159.                 //尽量少用延时,如果事件处理耗时较长,应在main中执行
  160.                 if(softTimer_8s_flag == 1)
  161.                 {
  162.                         LED4=~LED4;
  163.                         Delayms(200);//模拟耗时事件
  164.                         softTimer_8s_flag = 0;
  165.                 }        
  166.         }
  167. }
复制代码
51hei截图20191102225852.png

程序及仿真: 51单片机软件定时器测试.7z (48.06 KB, 下载次数: 109)

评分

参与人数 1黑币 +50 收起 理由
admin + 50 共享资料的黑币奖励!

查看全部评分

回复

使用道具 举报

ID:282431 发表于 2019-11-3 11:17 | 显示全部楼层
这种想法好,谢谢楼主
回复

使用道具 举报

ID:634898 发表于 2019-11-3 20:29 | 显示全部楼层
想法很好,学到了学到了
回复

使用道具 举报

ID:412814 发表于 2020-1-10 07:07 | 显示全部楼层
谢谢楼主分享,学习了。
回复

使用道具 举报

ID:475776 发表于 2020-1-22 11:30 | 显示全部楼层
怎么中定时器中断中调用定时器更新函数,这样会导致定时不准确。为什么不在主函数中调用?在定时器就是只更新计时变量。
回复

使用道具 举报

ID:475776 发表于 2020-1-22 11:36 | 显示全部楼层
如果采用在硬件定时器中断中变量计数的这种形式,需要添加防止计时变量溢出措施。如:变量自动清零同时校验延时时间是否超出定时范围
回复

使用道具 举报

ID:517794 发表于 2020-3-21 11:14 | 显示全部楼层
好厉害的样子!有点复杂了 看不是很懂
回复

使用道具 举报

ID:214461 发表于 2020-10-27 08:54 | 显示全部楼层
谢谢分享,楼主辛苦了!
回复

使用道具 举报

ID:229621 发表于 2022-1-7 08:30 | 显示全部楼层
经过压力测试了么?
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

手机版|小黑屋|51黑电子论坛 |51黑电子论坛6群 QQ 管理员QQ:125739409;技术交流QQ群281945664

Powered by 单片机教程网

快速回复 返回顶部 返回列表