久久久久久久999_99精品久久精品一区二区爱城_成人欧美一区二区三区在线播放_国产精品日本一区二区不卡视频_国产午夜视频_欧美精品在线观看免费

 找回密碼
 立即注冊

QQ登錄

只需一步,快速開始

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

完全由C語言編寫,高度可移植,超級牛逼的軟件定時器!

[復制鏈接]
跳轉到指定樓層
樓主
ID:68814 發表于 2019-9-14 19:57 | 只看該作者 回帖獎勵 |倒序瀏覽 |閱讀模式
帖子來自芯片之家公眾號

作為一個搞電子嵌入式的你來說,定時器相信絕對不會陌生,去翻看STM32的官方數據手冊,可能有四分之一的篇幅都在講定時器,可見定時器的重要性,定時器定時器嘛,最重要的功能還是定時。之前經常有師弟跟我抱怨,哎呀,這個單片機怎么只有2個定時器,我都不夠用了,怎么辦呀,然后一頭霧水,其實,只要有一個硬件定時器,在實際不是要求特別特別精準的場合,我們可以模擬很多個軟件定時器出來,今天,群主給大家分享一個完全由C語言編寫,高度可移植,超級牛逼的軟件定時器!

MultiTimer 是一個軟件定時器擴展模塊,可無限擴展你所需的定時器任務,取代傳統的標志位判斷方式, 更優雅更便捷地管理程序的時間觸發時序。一共有3個文件:
multi_timer.c--定時器c源文件;
multi_timer.h--定時器h頭文件;
main--用戶代碼文件。

我們直接上代碼:
  1. /*
  2. *     main.c
  3. *
  4. *      Created on: 20161229
  5. *      @Author   : 曉宇
  6. *      @version  :V1.0.0
  7. *
  8. *                        ,%%%%%%%%,
  9. *                      ,%%/\%%%%/\%%
  10. *                     ,%%%\c''''J/%%%
  11. *           %.        %%%%/ o  o \%%%
  12. *           `%%.      %%%%       |%%%
  13. *            `%%      `%%%%(__Y__)%%'
  14. *            //        ;%%%%`\-/%%%'
  15. *            ((      /   `%%%%%%%'
  16. *             \\     .'           |
  17. *              \\   /        \  | |
  18. *               \\/          ) | |
  19. *                \          /_ | |__
  20. *                (____________))))))) 攻城獅
  21. *
  22. */

  23. #include "multi_timer.h"

  24. struct Timer timer1;
  25. struct Timer timer2;

  26. void timer1_callback()
  27. {
  28.    printf("timer1 timeout!\r\n");
  29. }

  30. void timer2_callback()
  31. {
  32.    printf("timer2 timeout!\r\n");
  33. }

  34. int main()
  35. {
  36. timer_init(&timer1, timer1_callback, 1000, 1000); //1s loop
  37. timer_start(&timer1);

  38. timer_init(&timer2, timer2_callback, 50, 0); //50ms delay
  39. timer_start(&timer2);

  40. while(1) {
  41.      
  42.      timer_loop();
  43. }
  44. }

  45. void HAL_SYSTICK_Callback(void)
  46. {
  47.    timer_ticks(); //1ms ticks
  48. }
復制代碼


1、先申請一個定時器管理handle
  1. struct Timer timer1;
復制代碼


2、初始化定時器對象,注冊定時器回調處理函數,設置定時時間(ms),循環定時觸發時間

  1. timer_init(struct Timer* handle, void(*timeout_cb)(), uint32_t timeout, uint32_t repeat);
復制代碼


3.啟動定時器
  1. timer_start(&timer1);
復制代碼



4.設置1ms的硬件定時器循環調用 *timer_ticks()* 以提供時間基準

  1. void HAL_SYSTICK_Callback(void)
  2. {
  3.    timer_ticks();
  4. }
復制代碼



5.在主循環調用定時器后臺處理函數

  1. int main()
  2. {    ...
  3.    while(1) {
  4.        ...
  5.        timer_loop();
  6.    }
  7. }
復制代碼



上面,我們定義了兩個定時器,第一個以每1秒一次進行循環執行,第二個定時器50ms之后只執行一次,執行之后,可以通過相應的回調函數打印出來,里面就可以添加我們的定時任務了。怎么樣,是不是很簡單,只需要設置好定時基準,申請好定時器,初始化完畢,你想要的效果,是定時一次還是進行循環,都沒問題,多個定時器之間使用鏈表來實現,可以啟動暫停每個定時器,用起來之后,是不是很爽?具體的代碼,需要大家仔細去推敲其中的原理,當然直接用也是可以的。

部分源代碼展示:
  1. /*
  2. *     multi_timer.h
  3. *
  4. *      Created on: 20161229
  5. *      @Author   : 曉宇
  6. *      @version  :V1.0.0
  7. */

  8. #ifndef _MULTI_TIMER_H_
  9. #define _MULTI_TIMER_H_

  10. #include "stdint.h"

  11. typedef struct Timer {
  12.    uint32_t timeout;
  13.    uint32_t repeat;
  14.    void (*timeout_cb)(void);
  15.    struct Timer* next;
  16. }Timer;

  17. #ifdef __cplusplus  
  18. extern "C" {  
  19. #endif  

  20. void timer_init(struct Timer* handle, void(*timeout_cb)(), uint32_t timeout, uint32_t repeat);
  21. int  timer_start(struct Timer* handle);
  22. void timer_stop(struct Timer* handle);
  23. void timer_ticks(void);
  24. void timer_loop(void);

  25. // void timer_again(struct Timer* handle);
  26. // void timer_set_repeat(struct Timer* handle, uint32_t repeat);

  27. #ifdef __cplusplus
  28. }
  29. #endif

  30. #endif/*
  31. *     multi_timer.c
  32. *
  33. *      Created on: 20161229
  34. *      @Author   : 曉宇
  35. *      @version  :V1.0.0
  36. */


  37. #include "multi_timer.h"

  38. //timer handle list head.
  39. static struct Timer* head_handle = NULL;

  40. //Timer ticks
  41. static uint32_t _timer_ticks = 0;

  42. /**
  43. * @brief  Initializes the timer struct handle.
  44. * @param  handle: the timer handle strcut.
  45. * @param  timeout_cb: timeout callback.
  46. * @param  repeat: repeat interval time.
  47. * @retval None
  48. */
  49. void timer_init(struct Timer* handle, void(*timeout_cb)(), uint32_t timeout, uint32_t repeat)
  50. {
  51. // memset(handle, sizeof(struct Timer), 0);
  52. handle->timeout_cb = timeout_cb;
  53. handle->timeout = _timer_ticks + timeout;
  54. handle->repeat = repeat;
  55. }

  56. /**
  57. * @brief  Start the timer work, add the handle into work list.
  58. * @param  btn: target handle strcut.
  59. * @retval 0: succeed. -1: already exist.
  60. */
  61. int timer_start(struct Timer* handle)
  62. {
  63. struct Timer* target = head_handle;
  64. while(target) {
  65.    if(target == handle) return -1;  //already exist.
  66.    target = target->next;
  67. }
  68. handle->next = head_handle;
  69. head_handle = handle;
  70. return 0;
  71. }

  72. /**
  73. * @brief  Stop the timer work, remove the handle off work list.
  74. * @param  handle: target handle strcut.
  75. * @retval None
  76. */
  77. void timer_stop(struct Timer* handle)
  78. {
  79. struct Timer** curr;
  80. for(curr = &head_handle; *curr; ) {
  81.    struct Timer* entry = *curr;
  82.    if (entry == handle) {
  83.      *curr = entry->next;
  84. //      free(entry);
  85.    } else
  86.      curr = &entry->next;
  87. }
  88. }

  89. /**
  90. * @brief  main loop.
  91. * @param  None.
  92. * @retval None
  93. */
  94. void timer_loop()
  95. {
  96. struct Timer* target;
  97. for(target=head_handle; target; target=target->next) {
  98.    if(_timer_ticks >= target->timeout) {
  99.      if(target->repeat == 0) {
  100.        timer_stop(target);
  101.      } else {
  102.        target->timeout = _timer_ticks + target->repeat;
  103.      }
  104.      target->timeout_cb();
  105.    }
  106. }
  107. }

  108. /**
  109. * @brief  background ticks, timer repeat invoking interval 1ms.
  110. * @param  None.
  111. * @retval None.
  112. */
  113. void timer_ticks()
  114. {
  115. _timer_ticks++;
  116. }
復制代碼

全部資料51hei下載地址:
MultiTimer.zip (2.26 KB, 下載次數: 47)

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

使用道具 舉報

沙發
ID:428438 發表于 2019-9-16 10:30 | 只看該作者
學習一下,給樓主點個贊。軟定時還是經常用到的。
回復

使用道具 舉報

板凳
ID:68814 發表于 2019-9-23 08:08 | 只看該作者
leo2002zhang 發表于 2019-9-16 10:30
學習一下,給樓主點個贊。軟定時還是經常用到的。

32用的比較方便,51直接用標志位去完成軟定時器功能了
回復

使用道具 舉報

地板
ID:130230 發表于 2020-9-10 00:13 | 只看該作者
把github地址貼出來好一點,這一個是有開源協議的。
回復

使用道具 舉報

5#
無效樓層,該帖已經被刪除
您需要登錄后才可以回帖 登錄 | 立即注冊

本版積分規則

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

Powered by 單片機教程網

快速回復 返回頂部 返回列表
主站蜘蛛池模板: 97av在线| 欧美一区二区三区的 | 亚洲精品18 | 99精品欧美一区二区三区综合在线 | 性天堂网| 亚洲免费视频播放 | 欧美日韩综合精品 | 日韩久久久久久久久久久 | 国产精品毛片久久久久久 | 天天玩夜夜操 | 视频一区在线观看 | 美日韩视频 | av黄色在线 | 日本h片在线观看 | 欧美日韩国产传媒 | aaaa网站| 99精品视频免费观看 | av特级毛片| 久久久久中文字幕 | 亚洲导航深夜福利涩涩屋 | 国产成人精品a视频一区www | 国产精品成人国产乱一区 | 日韩一区二区三区在线播放 | 国产三级日本三级 | 操操操日日日 | 中文av字幕 | 亚洲另类自拍 | 国产精品久久久久久52avav | 欧美精品成人影院 | 国产999精品久久久影片官网 | 国产高清视频 | 国产激情综合五月久久 | 久久久久久免费免费 | 成人黄色a| 国产欧美一区二区精品久导航 | 亚洲一区二区不卡在线观看 | 国产精品高清一区二区 | 天天操操| 亚洲成av人片在线观看无码 | a a毛片| 大伊人久久 |