色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫(xiě)文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

定時(shí)器原理以及一般定時(shí)器實(shí)現(xiàn)的方式

開(kāi)關(guān)電源芯片 ? 來(lái)源:Linux內(nèi)核那些事 ? 作者:Linux內(nèi)核那些事 ? 2021-08-14 11:15 ? 次閱讀

定時(shí)器原理一般定時(shí)器實(shí)現(xiàn)的方式有以下幾種:

基于排序鏈表方式:

通過(guò)排序鏈表來(lái)保存定時(shí)器,由于鏈表是排序好的,所以獲取最小(最早到期)的定時(shí)器的時(shí)間復(fù)雜度為 O(1)。但插入需要遍歷整個(gè)鏈表,所以時(shí)間復(fù)雜度為 O(n)。如下圖:

基于最小堆方式:

通過(guò)最小堆來(lái)保存定時(shí)器,在最小堆中獲取最小定時(shí)器的時(shí)間復(fù)雜度為 O(1),但插入一個(gè)定時(shí)器的時(shí)間復(fù)雜度為 O(log n)。如下圖:

基于平衡二叉樹(shù)方式:

使用平衡二叉樹(shù)(如紅黑樹(shù))保存定時(shí)器,在平衡二叉樹(shù)中獲取最小定時(shí)器的時(shí)間復(fù)雜度為 O(log n)(也可以通過(guò)緩存最小值的方法來(lái)達(dá)到 O(1)),而插入一個(gè)定時(shí)器的時(shí)間復(fù)雜度為 O(log n)。如下圖:

時(shí)間輪:

但對(duì)于Linux這種對(duì)定時(shí)器依賴性比較高(網(wǎng)絡(luò)子模塊的TCP協(xié)議使用了大量的定時(shí)器)的操作系統(tǒng)來(lái)說(shuō),以上的數(shù)據(jù)結(jié)構(gòu)都是不能滿足要求的。所以Linux使用了效率更高的定時(shí)器算法:時(shí)間輪。

時(shí)間輪 類似于日常生活的時(shí)鐘

日常生活的時(shí)鐘,每當(dāng)秒針轉(zhuǎn)一圈時(shí),分針就會(huì)走一格,而分針走一圈時(shí),時(shí)針就會(huì)走一格。而時(shí)間輪的實(shí)現(xiàn)方式與時(shí)鐘類似,就是把到期時(shí)間當(dāng)成一個(gè)輪,然后把定時(shí)器掛在這個(gè)輪子上面,每當(dāng)時(shí)間走一秒就移動(dòng)時(shí)針,并且執(zhí)行那個(gè)時(shí)針上的定時(shí)器。

一般的定時(shí)器范圍為一個(gè)32位整型的大小,也就是 0 ~ 4294967295,如果通過(guò)一個(gè)數(shù)組來(lái)存儲(chǔ)的話,就需要一個(gè)元素個(gè)數(shù)為4294967296的數(shù)組,非常浪費(fèi)內(nèi)存。這個(gè)時(shí)候就可以通過(guò)類似于時(shí)鐘的方式:通過(guò)多級(jí)數(shù)組來(lái)存儲(chǔ)。

時(shí)鐘通過(guò)時(shí)分秒來(lái)進(jìn)行分級(jí),當(dāng)然我們也可以這樣,但對(duì)于計(jì)算機(jī)來(lái)說(shuō),時(shí)分秒的分級(jí)不太友好,所以Linux內(nèi)核中,對(duì)32位整型分為5個(gè)級(jí)別,第一個(gè)等級(jí)存儲(chǔ)0 ~ 255秒 的定時(shí)器,第二個(gè)等級(jí)為 256秒 ~ 256*64秒,第三個(gè)等級(jí)為 256*64秒 ~ 256*64*64秒,第四個(gè)等級(jí)為 256*64*64秒 ~ 256*64*64*64秒,第五個(gè)等級(jí)為 256*64*64*64秒 ~ 256*64*64*64*64秒。

注意:第二級(jí)至第五級(jí)數(shù)組的第一個(gè)槽是不掛任何定時(shí)器的。

每級(jí)數(shù)組上面都有一個(gè)指針,指向當(dāng)前要執(zhí)行的定時(shí)器。每當(dāng)時(shí)間走一秒,Linux首先會(huì)移動(dòng)第一級(jí)的指針,然后執(zhí)行當(dāng)前位置上的定時(shí)器。當(dāng)指針變?yōu)?時(shí),會(huì)移動(dòng)下一級(jí)的指針,并把該位置上的定時(shí)器重新計(jì)算一次并且插入到時(shí)間輪中,其他級(jí)如此類推。

當(dāng)要執(zhí)行到期的定時(shí)器只需要移動(dòng)第一級(jí)數(shù)組上的指針并且執(zhí)行該位置上的定時(shí)器列表即可,所以時(shí)間復(fù)雜度為 O(1),而插入一個(gè)定時(shí)器也很簡(jiǎn)單,先計(jì)算定時(shí)器的過(guò)期時(shí)間范圍在哪一級(jí)數(shù)組上,并且連接到該位置上的鏈表即可,時(shí)間復(fù)雜度也是 O(1)。

Linux時(shí)間輪的實(shí)現(xiàn)那么接下來(lái)我們看看Linux內(nèi)核是怎么實(shí)現(xiàn)時(shí)間輪算法的。

定義五個(gè)等級(jí)的數(shù)組

#define TVN_BITS 6#define TVR_BITS 8#define TVN_SIZE (1 《《 TVN_BITS) // 64#define TVR_SIZE (1 《《 TVR_BITS) // 256#define TVN_MASK (TVN_SIZE - 1)#define TVR_MASK (TVR_SIZE - 1)struct timer_vec {

int index;

struct list_head vec[TVN_SIZE];

};

struct timer_vec_root {

int index;

struct list_head vec[TVR_SIZE];

};

static struct timer_vec tv5;static struct timer_vec tv4;static struct timer_vec tv3;static struct timer_vec tv2;static struct timer_vec_root tv1;void init_timervecs (void)

{

int i;

for (i = 0; i 《 TVN_SIZE; i++) {

INIT_LIST_HEAD(tv5.vec + i);

INIT_LIST_HEAD(tv4.vec + i);

INIT_LIST_HEAD(tv3.vec + i);

INIT_LIST_HEAD(tv2.vec + i);

}

for (i = 0; i 《 TVR_SIZE; i++)

INIT_LIST_HEAD(tv1.vec + i);

}

上面的代碼定義第一級(jí)數(shù)組為 timer_vec_root 類型,其 index 成員是當(dāng)前要執(zhí)行的定時(shí)器指針(對(duì)應(yīng) vec 成員的下標(biāo)),而 vec 成員是一個(gè)鏈表數(shù)組,數(shù)組元素個(gè)數(shù)為256,每個(gè)元素上保存了該秒到期的定時(shí)器列表,其他等級(jí)的數(shù)組類似。

插入定時(shí)器

static inline void internal_add_timer(struct timer_list *timer)

{

/*

* must be cli-ed when calling this

*/

unsigned long expires = timer-》expires;

unsigned long idx = expires - timer_jiffies;

struct list_head * vec;

if (idx 《 TVR_SIZE) { // 0 ~ 255

int i = expires & TVR_MASK;

vec = tv1.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + TVN_BITS)) { // 256 ~ 16191

int i = (expires 》》 TVR_BITS) & TVN_MASK;

vec = tv2.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + 2 * TVN_BITS)) {

int i = (expires 》》 (TVR_BITS + TVN_BITS)) & TVN_MASK;

vec = tv3.vec + i;

} else if (idx 《 1 《《 (TVR_BITS + 3 * TVN_BITS)) {

int i = (expires 》》 (TVR_BITS + 2 * TVN_BITS)) & TVN_MASK;

vec = tv4.vec + i;

} else if ((signed long) idx 《 0) {

/* can happen if you add a timer with expires == jiffies,

* or you set a timer to go off in the past

*/

vec = tv1.vec + tv1.index;

} else if (idx 《= 0xffffffffUL) {

int i = (expires 》》 (TVR_BITS + 3 * TVN_BITS)) & TVN_MASK;

vec = tv5.vec + i;

} else {

/* Can only get here on architectures with 64-bit jiffies */

INIT_LIST_HEAD(&timer-》list);

return;

}

/*

* 添加到鏈表中

*/

list_add(&timer-》list, vec-》prev);

}

internal_add_timer() 函數(shù)的主要工作是計(jì)算定時(shí)器到期時(shí)間所屬的等級(jí)范圍,然后把定時(shí)器添加到鏈表中。

執(zhí)行到期的定時(shí)器

static inline void cascade_timers(struct timer_vec *tv)

{

/* cascade all the timers from tv up one level */

struct list_head *head, *curr, *next;

head = tv-》vec + tv-》index;

curr = head-》next;

/*

* We are removing _all_ timers from the list, so we don‘t have to

* detach them individually, just clear the list afterwards.

*/

while (curr != head) {

struct timer_list *tmp;

tmp = list_entry(curr, struct timer_list, list);

next = curr-》next;

list_del(curr);

internal_add_timer(tmp);

curr = next;

}

INIT_LIST_HEAD(head);

tv-》index = (tv-》index + 1) & TVN_MASK;

}

static inline void run_timer_list(void)

{

spin_lock_irq(&timerlist_lock);

while ((long)(jiffies - timer_jiffies) 》= 0) {

struct list_head *head, *curr;

if (!tv1.index) { // 完成了一個(gè)輪回, 移動(dòng)下一個(gè)單位的定時(shí)器

int n = 1;

do {

cascade_timers(tvecs[n]);

} while (tvecs[n]-》index == 1 && ++n 《 NOOF_TVECS);

}

repeat:

head = tv1.vec + tv1.index;

curr = head-》next;

if (curr != head) {

struct timer_list *timer;

void (*fn)(unsigned long);

unsigned long data;

timer = list_entry(curr, struct timer_list, list);

fn = timer-》function;

data= timer-》data;

detach_timer(timer);

timer-》list.next = timer-》list.prev = NULL;

timer_enter(timer);

spin_unlock_irq(&timerlist_lock);

fn(data);

spin_lock_irq(&timerlist_lock);

timer_exit();

goto repeat;

}

++timer_jiffies;

tv1.index = (tv1.index + 1) & TVR_MASK;

}

spin_unlock_irq(&timerlist_lock);

}

執(zhí)行到期的定時(shí)器主要通過(guò) run_timer_list() 函數(shù)完成,該函數(shù)首先比較當(dāng)前時(shí)間與最后一次運(yùn)行 run_timer_list() 函數(shù)時(shí)間的差值,然后循環(huán)這個(gè)差值的次數(shù),并執(zhí)行當(dāng)前指針位置上的定時(shí)器。

每循環(huán)一次對(duì)第一級(jí)數(shù)組指針進(jìn)行加一操作,當(dāng)?shù)谝患?jí)數(shù)組指針變?yōu)?(即所有定時(shí)器都執(zhí)行完),那么就移動(dòng)下一個(gè)等級(jí)的指針,并把該位置上的定時(shí)器重新計(jì)算插入到時(shí)間輪中,重新計(jì)算定時(shí)器通過(guò) cascade_timers() 函數(shù)實(shí)現(xiàn)。

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫(xiě)或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 計(jì)算機(jī)
    +關(guān)注

    關(guān)注

    19

    文章

    7488

    瀏覽量

    87854
  • 定時(shí)器
    +關(guān)注

    關(guān)注

    23

    文章

    3246

    瀏覽量

    114721
  • TCP協(xié)議
    +關(guān)注

    關(guān)注

    1

    文章

    91

    瀏覽量

    12070

原文標(biāo)題:一文讀懂:Linux定時(shí)器實(shí)現(xiàn)

文章出處:【微信號(hào):gh_3980db2283cd,微信公眾號(hào):開(kāi)關(guān)電源芯片】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    定時(shí)器的工作方式介紹

    定時(shí)器是計(jì)算機(jī)和嵌入式系統(tǒng)中常見(jiàn)的種硬件模塊,用于實(shí)現(xiàn)定時(shí)和計(jì)數(shù)功能。定時(shí)器的工作方式通常由
    的頭像 發(fā)表于 07-12 10:29 ?904次閱讀

    定時(shí)器相關(guān)的寄存有哪些類型

    在微控制編程中,定時(shí)器種非常常見(jiàn)的功能模塊,用于實(shí)現(xiàn)各種定時(shí)和計(jì)數(shù)功能。定時(shí)器的工作原理是
    的頭像 發(fā)表于 07-12 10:25 ?904次閱讀

    鴻蒙開(kāi)發(fā)系統(tǒng)基礎(chǔ)能力:Timer定時(shí)器

    設(shè)置個(gè)定時(shí)器,該定時(shí)器定時(shí)器到期后執(zhí)行個(gè)函數(shù)。
    的頭像 發(fā)表于 06-28 11:33 ?986次閱讀
    鴻蒙開(kāi)發(fā)系統(tǒng)基礎(chǔ)能力:Timer<b class='flag-5'>定時(shí)器</b>

    三菱plc如何顯示定時(shí)器時(shí)間

    三菱PLC(Programmable Logic Controller,可編程邏輯控制)是種廣泛應(yīng)用于工業(yè)自動(dòng)化領(lǐng)域的設(shè)備。定時(shí)器是PLC中的種基本功能,用于
    的頭像 發(fā)表于 06-20 11:10 ?1847次閱讀

    如何實(shí)現(xiàn)個(gè)軟件定時(shí)器

    在Linux,uC/OS,F(xiàn)reeRTOS等操作系統(tǒng)中,都帶有軟件定時(shí)器,原理大同小異。典型的實(shí)現(xiàn)方法是:通過(guò)個(gè)硬件定時(shí)器產(chǎn)生固定的時(shí)鐘節(jié)拍,每次硬件
    的頭像 發(fā)表于 04-29 11:00 ?632次閱讀

    s7200定時(shí)器的五種故障介紹

    定時(shí)器或CPU故障:如果定時(shí)器本身或PLC的CPU出現(xiàn)故障,也可能導(dǎo)致定時(shí)器無(wú)法復(fù)位。此時(shí),需要檢查定時(shí)器和CPU的工作狀態(tài),確保其正常運(yùn)行。
    的頭像 發(fā)表于 04-03 17:08 ?2421次閱讀

    ?PLC定時(shí)器介紹

    定時(shí)器是PLC中重要的編程元件,是累計(jì)時(shí)間增量的內(nèi)部器件。大部分自動(dòng)控制領(lǐng)域都需要定時(shí)器進(jìn)行延時(shí)控制,靈活地使用定時(shí)器可以編制出復(fù)雜的控制程序。
    發(fā)表于 03-22 12:36 ?2330次閱讀
    ?PLC<b class='flag-5'>定時(shí)器</b>介紹

    使用555定時(shí)器的可調(diào)雙定時(shí)器電路

    定時(shí)器 IC 555 是最通用和最常用的 IC 之,因?yàn)樗膽?yīng)用范圍更廣,如 PWM放大器、延遲定時(shí)器、開(kāi)關(guān)電路、占空比選擇、時(shí)鐘脈沖發(fā)生
    的頭像 發(fā)表于 02-25 15:16 ?2187次閱讀
    使用555<b class='flag-5'>定時(shí)器</b>的可調(diào)雙<b class='flag-5'>定時(shí)器</b>電路

    定時(shí)器原理能控制馬達(dá)嗎為什么

    定時(shí)器原理可以用于控制馬達(dá)。馬達(dá)是種將電能轉(zhuǎn)換為機(jī)械能的設(shè)備,通常由電動(dòng)機(jī)和傳動(dòng)裝置組成。定時(shí)器種電子設(shè)備,用來(lái)生成和計(jì)時(shí)精確而穩(wěn)定的時(shí)間信號(hào)。通過(guò)將
    的頭像 發(fā)表于 01-23 15:21 ?662次閱讀

    555定時(shí)器的基本功能 555定時(shí)器的工作原理及其應(yīng)用

    555定時(shí)器種非常常見(jiàn)和常用的集成電路,它具有廣泛的應(yīng)用領(lǐng)域,例如計(jì)時(shí)、頻率分頻、脈沖寬度調(diào)制等。本文將詳細(xì)介紹555定時(shí)器的基本功能、工作原理以及應(yīng)用。
    的頭像 發(fā)表于 01-18 11:12 ?1.5w次閱讀

    時(shí)間定時(shí)器開(kāi)關(guān)怎樣設(shè)置時(shí)間

    自己,或者設(shè)定一定時(shí)間后自動(dòng)關(guān)閉電視等。 設(shè)置時(shí)間定時(shí)器的方法和步驟可能會(huì)因具體設(shè)備而有所差異,下面將以一般常見(jiàn)的時(shí)間定時(shí)器為例,逐步介紹設(shè)置時(shí)間
    的頭像 發(fā)表于 01-16 16:32 ?4560次閱讀

    AWTK 開(kāi)源串口屏開(kāi)發(fā)(6) - 定時(shí)器的用法

    定時(shí)器是個(gè)常用的功能,AWTK串口屏提供了豐富的定時(shí)器函數(shù),用于定時(shí)器的啟動(dòng)、停止、暫停、恢復(fù)、修改和重置等功能,本文以計(jì)時(shí)的例子來(lái)介紹定時(shí)器
    的頭像 發(fā)表于 01-13 08:24 ?569次閱讀
    AWTK 開(kāi)源串口屏開(kāi)發(fā)(6) - <b class='flag-5'>定時(shí)器</b>的用法

    AT32 定時(shí)器配置中pr和div的作用

    AT32定時(shí)器是51系列單片機(jī)中的定時(shí)器,可以實(shí)現(xiàn)多種定時(shí)功能。在AT32定時(shí)器中,pr和d
    的頭像 發(fā)表于 01-08 10:12 ?1297次閱讀

    單片機(jī)定時(shí)器的用法

    本章以CW32通用定時(shí)器為例介紹單片機(jī)定時(shí)器的用法。
    的頭像 發(fā)表于 01-04 10:37 ?1390次閱讀
    單片機(jī)<b class='flag-5'>定時(shí)器</b>的用法

    51單片機(jī)定時(shí)器定時(shí)1秒程序流水燈結(jié)果分析

    流水燈的設(shè)計(jì)原理、實(shí)現(xiàn)步驟和結(jié)果分析。 、設(shè)計(jì)原理 在51單片機(jī)中,定時(shí)器常用的工作方式有兩種:定時(shí)器模式和計(jì)數(shù)
    的頭像 發(fā)表于 12-26 14:57 ?7169次閱讀
    主站蜘蛛池模板: 嫩草影院一区| 久久久国产精品免费A片蜜臀| 伊人国产视频| 免费完整版观看| 囯产精品麻豆巨作久久| 亚洲精品午睡沙发系列| 麻豆E奶女教师国产精品| 超大BBWWW| 亚洲精品天堂自在久久77| 免费亚洲视频在线观看| 国产超嫩一线天在线播放 | 野花视频在线观看免费| 男人国产AV天堂WWW麻豆| 国产国产乱老熟女视频网站97 | 黄色三级网址| 99成人在线视频| 无码国产色欲XXXX视频| 久久久久国产精品美女毛片| xnxx18美女| 亚洲精品视频在线观看视频| 欧美人与善交大片| 国产婷婷午夜无码A片| 97精品伊人久久大香线蕉app| 特级毛片s级全部免费| 久色乳综合思思在线视频| 东北女人奶大毛多水多| 一个人看的www视频动漫版| 热中文热国产热综合| 吉吉影音先锋av资源| YELLOW视频在线观看最新| 亚洲色噜噜狠狠网站| 日本久久频这里精品99| 久久国内精品视频| 俄罗斯mm| 91se在线看片国产免费观看| 无限资源日本2019版免费| 蜜芽无码亚洲资源网站| 国产品无码一区二区三区在线| 99国产精品白浆在线观看免费| 亚洲国产高清福利视频| 青青草原在线新免费|