找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

搜索
查看: 8378|回復: 8
打印 上一主題 下一主題
收起左側

51單片機基于一個定時器實現(xiàn)多個軟件定時器

  [復制鏈接]
跳轉到指定樓層
樓主
ID:171060 發(fā)表于 2019-11-2 23:31 | 只看該作者 |只看大圖 回帖獎勵 |倒序瀏覽 |閱讀模式
本帖最后由 lw1997 于 2019-11-2 23:33 編輯

軟件定時器是用程序模擬出來的定時器,可以由一個硬件定時器模擬出成千上萬個軟件定時器,這樣程序在需要使用較多定時器的時候就不會受限于硬件資源的不足,這是軟件定時器的一個優(yōu)點,即數(shù)量不受限制。但由于軟件定時器是通過程序實現(xiàn)的,其運行和維護都需要耗費一定的CPU資源,同時精度也相對硬件定時器要差一些。在RTOS等操作系統(tǒng)中,都帶有軟件定時器,原理大同小異。典型的實現(xiàn)方法是:通過一個硬件定時器產生固定的時鐘節(jié)拍,每次硬件定時器中斷到,就對一個全局的時間標記加一,每個軟件定時器都保存著到期時間,程序需要定期掃描所有運行中的軟件定時器,將各個到期時間與全局時鐘標記做比較,以判斷對應軟件定時器是否到期,到期則執(zhí)行相應的回調函數(shù),并關閉該定時器
程序如下,通過一個硬件定時器產生固定的時鐘節(jié)拍,用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                //定時器個數(shù)

  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 函數(shù)指針類型

  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;           //狀態(tài)
  28.         unsigned char mode;            //模式
  29.         unsigned int  period;          //定時周期
  30.         unsigned int  count;                   //定時計數(shù)用
  31.         pFun callback;                                   //定時器回調函數(shù)
  32. } softTimer;

  33. //定時器結構數(shù)組
  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. //清除定時器狀態(tài)(用于執(zhí)行事件后手動清除定時器狀態(tài))
  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. //獲取定時器狀態(tài)
  64. unsigned char softTimer_GetState(unsigned int id)
  65. {
  66.         return softTimerList[id].state;
  67. }

  68. void softTimer_Update(void)        //更新定時器狀態(tài),在硬件定時器中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()        //主函數(shù)
  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中執(zhí)行
  160.                 if(softTimer_8s_flag == 1)
  161.                 {
  162.                         LED4=~LED4;
  163.                         Delayms(200);//模擬耗時事件
  164.                         softTimer_8s_flag = 0;
  165.                 }        
  166.         }
  167. }
復制代碼


程序及仿真: 51單片機軟件定時器測試.7z (48.06 KB, 下載次數(shù): 109)


評分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎勵!

查看全部評分

分享到:  QQ好友和群QQ好友和群 QQ空間QQ空間 騰訊微博騰訊微博 騰訊朋友騰訊朋友
收藏收藏11 分享淘帖 頂 踩
回復

使用道具 舉報

沙發(fā)
ID:282431 發(fā)表于 2019-11-3 11:17 | 只看該作者
這種想法好,謝謝樓主
回復

使用道具 舉報

板凳
ID:634898 發(fā)表于 2019-11-3 20:29 | 只看該作者
想法很好,學到了學到了
回復

使用道具 舉報

地板
ID:412814 發(fā)表于 2020-1-10 07:07 | 只看該作者
謝謝樓主分享,學習了。
回復

使用道具 舉報

5#
ID:475776 發(fā)表于 2020-1-22 11:30 | 只看該作者
怎么中定時器中斷中調用定時器更新函數(shù),這樣會導致定時不準確。為什么不在主函數(shù)中調用?在定時器就是只更新計時變量。
回復

使用道具 舉報

6#
ID:475776 發(fā)表于 2020-1-22 11:36 | 只看該作者
如果采用在硬件定時器中斷中變量計數(shù)的這種形式,需要添加防止計時變量溢出措施。如:變量自動清零同時校驗延時時間是否超出定時范圍
回復

使用道具 舉報

7#
ID:517794 發(fā)表于 2020-3-21 11:14 | 只看該作者
好厲害的樣子!有點復雜了 看不是很懂
回復

使用道具 舉報

8#
ID:214461 發(fā)表于 2020-10-27 08:54 | 只看該作者
謝謝分享,樓主辛苦了!
回復

使用道具 舉報

9#
ID:229621 發(fā)表于 2022-1-7 08:30 | 只看該作者
經過壓力測試了么?
回復

使用道具 舉報

您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規(guī)則

小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術交流QQ群281945664

Powered by 單片機教程網

快速回復 返回頂部 返回列表