《電子技術應用》
您所在的位置:首頁 > 可編程邏輯 > 其他 > Linux教學——linux編程之經典多級時間輪定時器(C語言版)

Linux教學——linux編程之經典多級時間輪定時器(C語言版)

2022-11-07
來源:電子技術應用專欄作家 一口Linux

  一. 多級時間輪實現框架

微信截圖_20221107164603.png

  上圖是5個時間輪級聯的效果圖。中間的大輪是工作輪,只有在它上的任務才會被執行;其他輪上的任務時間到后遷移到下一級輪上,他們最終都會遷移到工作輪上而被調度執行。

  多級時間輪的原理也容易理解:就拿時鐘做說明,秒針轉動一圈分針轉動一格;分針轉動一圈時針轉動一格;同理時間輪也是如此:當低級輪轉動一圈時,高一級輪轉動一格,同時會將高一級輪上的任務重新分配到低級輪上。從而實現了多級輪級聯的效果。

  1.1 多級時間輪對象

  微信截圖_20221107164808.png

  多級時間輪應該至少包括以下內容:

  每一級時間輪對象

  輪子上指針的位置 關于輪子上指針的位置有一個比較巧妙的辦法:那就是位運算。比如定義一個無符號整型的數:

  微信截圖_20221107164832.png

  通過獲取當前的系統時間便可以通過位操作轉換為時間輪上的時間,通過與實際時間輪上的時間作比較,從而確定時間輪要前進調度的時間,進而操作對應時間輪槽位對應的任務。

  為什么至少需要這兩個成員呢?

  定義多級時間輪,首先需要明確的便是級聯的層數,也就是說需要確定有幾個時間輪。

  輪子上指針位置,就是當前時間輪運行到的位置,它與真實時間的差便是后續時間輪需要調度執行,它們的差值是時間輪運作起來的驅動力。

  多級時間輪對象的定義

 微信截圖_20221107164856.png

  1.2 時間輪對象

  微信截圖_20221107164914.png

  我們知道每一個輪子實際上都是一個哈希表,上面我們只是實例化了五個輪子的對象,但是五個輪子具體包含什么,有幾個槽位等等沒有明確(即struct tvec和struct tvec_root)。

微信截圖_20221107164932.png

  此外,每一個時間輪都是哈希表,因此它的類型應該至少包含兩個指針域來實現雙向鏈表的功能。這里我們為了方便使用通用的struct list_head的雙向鏈表結構。

  1.3 定時任務對象

  微信截圖_20221107165005.png

  定時器的主要工作是為了在未來的特定時間完成某項任務,而這個任務經常包含以下內容:

  任務的處理邏輯(回調函數)

  任務的參數

  雙向鏈表節點

  到時時間

  定時任務對象的定義

 微信截圖_20221107165026.png

  在時間輪上的效果圖:

  微信截圖_20221107165050.png

  1.4 雙向鏈表

  在時間輪上我們采用雙向鏈表的數據類型。采用雙向鏈表的除了操作上比單鏈表復雜,多占一個指針域外沒有其他不可接收的問題。而多占一個指針域在今天大內存的時代明顯不是什么問題。至于雙向鏈表操作的復雜性,我們可以通過使用通用的struct list結構來解決,因為雙向鏈表有眾多的標準操作函數,我們可以通過直接引用list.h頭文件來使用他們提供的接口。

  struct list可以說是一個萬能的雙向鏈表操作框架,我們只需要在自定義的結構中定義一個struct list對象即可使用它的標準操作接口。同時它還提供了一個類似container_of的接口,在應用層一般叫做list_entry,因此我們可以很方便的通過struct list成員找到自定義的結構體的起始地址。

  關于應用層的log.h, 我將在下面的代碼中附上該文件。如果需要內核層的實現,可以直接從linux源碼中獲取。

  1.5 聯結方式

  多級時間輪效果圖:

  微信截圖_20221107165120.png

  二. 多級時間輪C語言實現

  2.1 雙向鏈表頭文件: list.h

  提到雙向鏈表,很多的源碼工程中都會實現一系列的統一的雙向鏈表操作函數。它們為雙向鏈表封裝了統計的接口,使用者只需要在自定義的結構中添加一個struct list_head結構,然后調用它們提供的接口,便可以完成雙向鏈表的所有操作。這些操作一般都在list.h的頭文件中實現。Linux源碼中也有實現(內核態的實現)。他們實現的方式基本完全一樣,只是實現的接口數量和功能上稍有差別??梢哉f這個list.h文件是學習操作雙向鏈表的不二選擇,它幾乎實現了所有的操作:增、刪、改、查、遍歷、替換、清空等等。這里我拼湊了一個源碼中的log.h函數,終于湊夠了多級時間輪中使用到的接口。

微信截圖_20221107165224.png

微信截圖_20221107165254.png

微信截圖_20221107165330.png

微信截圖_20221107165352.png

微信截圖_20221107165433.png

  這里面一般會用到一個重要實現:container_of, 它的原理這里不敘述

  2.2 調試信息頭文件: log.h

  這個頭文件實際上不是必須的,我只是用它來添加調試信息(代碼中的errlog(), log()都是log.h中的宏函數)。它的效果是給打印的信息加上顏色,效果如下:

  微信截圖_20221107165500.png

  log.h的代碼如下:

  微信截圖_20221107165530.png

  2.3 時間輪代碼: timewheel.c

微信截圖_20221107165659.png

微信截圖_20221107165725.png微信截圖_20221107165755.png微信截圖_20221107165839.png微信截圖_20221107165911.png微信截圖_20221107165944.png

微信截圖_20221107170011.png

微信截圖_20221107170038.png微信截圖_20221107170206.png

微信截圖_20221107170332.png

微信截圖_20221107170350.png

  2.4 編譯運行

  微信截圖_20221107170433.png

  從結果可以看出:如果添加的定時任務是比較耗時的操作,那么后續的任務也會被阻塞,可能一直到超時,甚至一直阻塞下去,這個取決于當前任務是否耗時。

  這個理論上是絕不能接受的:一個任務不應該也不能去影響其他的任務吧。但是目前沒有對此問題進行改進和完善,以后有機會再繼續完善吧。


  更多信息可以來這里獲取==>>電子技術應用-AET<<

微信圖片_20210517164139.jpg

微信圖片_20220701092006.jpg

電子技術應用專欄作家  一口Linux

原文鏈接:https://mp.weixin.qq.com/s/pmHS5IlW1gfQSyeIYu3IGQ

本站內容除特別聲明的原創文章之外,轉載內容只為傳遞更多信息,并不代表本網站贊同其觀點。轉載的所有的文章、圖片、音/視頻文件等資料的版權歸版權所有權人所有。本站采用的非本站原創文章及圖片等內容無法一一聯系確認版權者。如涉及作品內容、版權和其它問題,請及時通過電子郵件或電話通知我們,以便迅速采取適當措施,避免給雙方造成不必要的經濟損失。聯系電話:010-82306118;郵箱:aet@chinaaet.com。
主站蜘蛛池模板: 免费国产不卡午夜福在线观看 | 日本精品久久久久中文字幕 1 | 国产成人综合网在线观看 | 日韩视频免费在线观看 | 午夜毛片网站 | 九九亚洲| 人人爽人人爽人人片c | 国产91色综合久久免费分享 | 热久久国产 | 天天躁夜夜躁狠狠躁躁88 | 国产视频高清在线 | 九九视频在线免费观看 | 欧美精品成人久久网站 | 久久99久久99精品免观看 | 亚洲成a人片77777kkkk | 李丽珍电影免费观看全集 | 成人精品视频网站 | 欧美一区精品 | 久久国产精品亚洲一区二区 | 日韩精品欧美视频 | 五月婷婷丁香六月 | 找国产毛片看 | 四虎影视www四虎免费 | 日韩写真集福利视频 | 午夜精品久久久久久 | 成人亚洲性情网站www在线观看 | 欧美 日韩 中文 | 77777影视在线观看免费视频 | 国产成人女人在线视频观看 | 国产成人综合在线 | 91精品国产综合久久精品 | 国内在线视频 | 五月天婷婷在线观看高清 | 2020久久国产最新免费观看 | 毛片福利 | 香蕉视频在线精品 | 91精品国产91久久久久久最新 | 男男羞羞视频网站国产 | 最新激情网 | 狠狠五月婷婷 | 久久精品视频6 |