《電子技術應用》
您所在的位置:首頁 > 可編程邏輯 > 其他 > 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源碼中也有實現(內核態的實現)。他們實現的方式基本完全一樣,只是實現的接口數量和功能上稍有差別。可以說這個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。
主站蜘蛛池模板: 色天使久久综合给合久久97色 | 亚洲午夜精品久久久久久抢 | 日本黄色大片视频 | 久久91精品久久久久久水蜜桃 | 成年人毛片网站 | 久久加久久 | 青在线视频 | 免费国产成人手机在线观看 | 国产免费拔擦拔擦8x | 免费一级录像 | 亚洲欧美中文v日韩v在线 | 欧美一级二级三级视频 | 激情五月激情综合网 | 成年人在线免费看视频 | 五月丁香六月综合缴清无码 | 欧美人成在线观看ccc36 | 国产精品日韩欧美 | 看片在线 | 国产一级高清视频在线 | 91精选视频在线观看 | 一级毛片成人免费看a | 国产麻豆视频免费观看 | a级黄色免费 | 男女性刺激爽爽免费视频 | 一级毛片视频播放 | 一级做a爱片在线播放 | 国产69精品久久久久9999 | 国产欧美精品一区二区三区四区 | 一本色道久久88 | 三级伦理影院 | 91精品小视频 | 国产成人高清在线 | 欧美日韩精品一区二区在线线 | 日韩成人片 | www.爱色.com| 免费黄色在线网址 | 亚洲欧美日韩精品久久亚洲区色播 | 最近中文字幕视频 | 日本xxxx18vr| 亚洲综合伦理一区 | 欧美日韩在线视频专区免费 |