Linux 定時器(四) 時間堆

概 述

在之前文章中實現(xiàn)的定時器跷跪,都是以固定的頻率調(diào)用心搏函數(shù)tick()万俗,從而處理到期的定時器任務(wù)康二。時間堆則采用另一種思路:將所有定時器中到期時間最小的值作為心搏間隔嫡纠。當(dāng)調(diào)用心搏函數(shù)時,此定時器任務(wù)必定到期丸冕,處理完此定時器任務(wù)后耽梅,再從所有定時器中到期時間最小的那個,將此到期時間設(shè)為下一次的心搏間隔胖烛,如此反復(fù)眼姐,實現(xiàn)定時。

實 現(xiàn)

從時間堆定時器的設(shè)計思路中不難看出佩番,使用小根堆結(jié)構(gòu)最適合處理這種方案众旗。我們通過對STL中priority_queue<>的封裝來實現(xiàn)時間堆(有關(guān)優(yōu)先級隊列的知識這里不做展示了)。書中則是手寫了一個基于數(shù)組的小根堆來實現(xiàn)時間堆趟畏,感興趣的可以自行學(xué)習(xí)下贡歧,這里也不再展示。示例代碼如下:

//TimerHeap.h
#include <netinet/in.h>
#include <functional>
#include <vector>
#include <queue>

#define BUFFER_SIZE 0xFFFF  //緩存區(qū)數(shù)據(jù)大小
#define TIMESLOT    30      //定時時間

class heap_timer; //前向聲明

// 客戶端數(shù)據(jù)
struct client_data {
    sockaddr_in address;
    int sockfd;
    char buf[BUFFER_SIZE];
    heap_timer *timer;
};

// 定時器類
class heap_timer {
public:
    heap_timer(int delay) {
        expire = time(nullptr) + delay;
    }
public:
    time_t expire; //定時器生效的絕對時間
    std::function<void(client_data *)> callBackFunc; //回調(diào)函數(shù)
    client_data *user_data; //用戶數(shù)據(jù)
};

struct cmp { //比較函數(shù)赋秀,實現(xiàn)小根堆
    bool operator () (const heap_timer* a, const heap_timer* b) {
        return a->expire > b->expire;
    }
};

class TimerHeap {
public:
    explicit TimerHeap();
    ~TimerHeap();

public:
    void AddTimer(heap_timer *timer); //添加定時器
    void DelTimer(heap_timer *timer); //刪除定時器
    void Tick(); //心搏函數(shù)
    heap_timer* Top();

private:
    std::priority_queue<heap_timer *, std::vector<heap_timer *>, cmp> m_timer_pqueue; //時間堆
};

//TimerHeap.cpp
#include "TimerHeap.h"

TimerHeap::TimerHeap() = default;

TimerHeap::~TimerHeap() {
    while (!m_timer_pqueue.empty()) {
        delete m_timer_pqueue.top();
        m_timer_pqueue.pop();
    }
}

void TimerHeap::AddTimer(heap_timer *timer) {
    if (!timer) return;
    m_timer_pqueue.emplace(timer);
}

void TimerHeap::DelTimer(heap_timer *timer) {
    if (!timer) return;
    // 將目標(biāo)定時器的回調(diào)函數(shù)設(shè)為空利朵,即延時銷毀
    // 減少刪除定時器的開銷,但會擴大優(yōu)先級隊列的大小
    timer->callBackFunc = nullptr;
}

void TimerHeap::Tick() {
    time_t cur = time(nullptr);
    while (!m_timer_pqueue.empty()) {
        heap_timer *timer = m_timer_pqueue.top();
        //堆頂定時器沒有到期猎莲,退出循環(huán)
        if (timer->expire > cur) break;
        //否則執(zhí)行堆頂定時器中的任務(wù)
        if (timer->callBackFunc) timer->callBackFunc(timer->user_data);
        m_timer_pqueue.pop();
    }
}

heap_timer *TimerHeap::Top() {
    if (m_timer_pqueue.empty()) return nullptr;
    else return m_timer_pqueue.top();
}

運 用

因為時間堆以最小到期時間作為心搏間隔绍弟,所以我們在服務(wù)器類中定義一個變量來記錄當(dāng)前是否在進(jìn)行定時任務(wù),從而來判斷是否觸發(fā)SIGALRM信號著洼,核心代碼如下:

// SIGALRM信號處理函數(shù)
void Server::TimerHandler() {
    m_timerHeap.Tick(); //調(diào)用時間堆的心搏函數(shù) Tick() 處理到期的任務(wù)
    heap_timer *tmp = nullptr;
    //判斷當(dāng)前是否在進(jìn)行定時任務(wù)晌柬,沒有則觸發(fā) SIGALRM 信號
    if (!s_isAlarm && (tmp = m_timerHeap.Top())) { 
        time_t delay = tmp->expire - time(nullptr);
        if (delay <= 0) delay = 1;
        alarm(delay); //觸發(fā) SIGALRM 信號
        s_isAlarm = true; //設(shè)置當(dāng)前已有定時任務(wù)在進(jìn)行
    }
}

// 定時器回調(diào)函數(shù)
void Server::TimerCallBack(client_data *user_data) {
    epoll_ctl(s_epollFd, EPOLL_CTL_DEL, user_data->sockfd, 0);
    if (user_data) {
        close(user_data->sockfd);
        s_clientsList.remove(user_data->sockfd);
        s_isAlarm = false; //設(shè)置當(dāng)前沒有進(jìn)行定時任務(wù)
        cout << "Server: close socket fd : " << user_data->sockfd << endl;
    }
}

void Server::Run() {
    //...
    while (!stop_server) {
        //... epoll_wait        

        for (int i = 0; i < ret; ++i) {
            int sockfd = events[i].data.fd;
            if (sockfd == sock_fd) { //處理新的客戶端連接
                //...accept
                
                //創(chuàng)建定時器
                heap_timer *timer = new heap_timer(TIMESLOT);
                timer->user_data = &m_user[conn_fd];
                timer->callBackFunc = TimerCallBack;
                m_timerHeap.AddTimer(timer);
                if (!s_isAlarm) { //當(dāng)前沒有進(jìn)行定時任務(wù)
                    s_isAlarm = true;
                    alarm(TIMESLOT); //觸發(fā) SIGALRM 信號
                }
                
            } else if ((sockfd == pipefd[0]) && (events[i].events & EPOLLIN)) { //處理信號
                //...
            } else if (events[i].events & EPOLLIN) { //處理客戶端數(shù)據(jù)
                //...
                heap_timer *timer = m_user[sockfd].timer;
                
                // 更新時間堆定時器
                if (timer) {
                    timer->expire += TIMESLOT;
                    // 更新定時器后姥份,需要設(shè)置當(dāng)前沒有進(jìn)行定時任務(wù)
                    // 從而在 TimerHandler() 中重新觸發(fā) SIGALRM 信號
                    s_isAlarm = false;
                }

            } else {
                //...
            }
        }
        //...處理定時任務(wù)
    }
    //... close
    return 0;
}

服務(wù)器程序的運行結(jié)果與之前實現(xiàn)的定時器類似,這里就不展示了年碘。
更多內(nèi)容,詳見GitHub:ChatRoomServer

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末展鸡,一起剝皮案震驚了整個濱河市屿衅,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌莹弊,老刑警劉巖涤久,帶你破解...
    沈念sama閱讀 211,123評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異忍弛,居然都是意外死亡响迂,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評論 2 384
  • 文/潘曉璐 我一進(jìn)店門细疚,熙熙樓的掌柜王于貴愁眉苦臉地迎上來蔗彤,“玉大人,你說我怎么就攤上這事疯兼∪欢簦” “怎么了?”我有些...
    開封第一講書人閱讀 156,723評論 0 345
  • 文/不壞的土叔 我叫張陵吧彪,是天一觀的道長待侵。 經(jīng)常有香客問我,道長姨裸,這世上最難降的妖魔是什么秧倾? 我笑而不...
    開封第一講書人閱讀 56,357評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮傀缩,結(jié)果婚禮上那先,老公的妹妹穿的比我還像新娘。我一直安慰自己扑毡,他們只是感情好胃榕,可當(dāng)我...
    茶點故事閱讀 65,412評論 5 384
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般性湿。 火紅的嫁衣襯著肌膚如雪噪径。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,760評論 1 289
  • 那天楔壤,我揣著相機與錄音,去河邊找鬼惯驼。 笑死蹲嚣,一個胖子當(dāng)著我的面吹牛递瑰,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播隙畜,決...
    沈念sama閱讀 38,904評論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼抖部,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了议惰?” 一聲冷哼從身側(cè)響起慎颗,我...
    開封第一講書人閱讀 37,672評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎言询,沒想到半個月后俯萎,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,118評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡运杭,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,456評論 2 325
  • 正文 我和宋清朗相戀三年夫啊,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片辆憔。...
    茶點故事閱讀 38,599評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡撇眯,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出躁愿,到底是詐尸還是另有隱情叛本,我是刑警寧澤,帶...
    沈念sama閱讀 34,264評論 4 328
  • 正文 年R本政府宣布彤钟,位于F島的核電站来候,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏逸雹。R本人自食惡果不足惜营搅,卻給世界環(huán)境...
    茶點故事閱讀 39,857評論 3 312
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望梆砸。 院中可真熱鬧转质,春花似錦、人聲如沸帖世。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,731評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽日矫。三九已至赂弓,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間哪轿,已是汗流浹背盈魁。 一陣腳步聲響...
    開封第一講書人閱讀 31,956評論 1 264
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留窃诉,地道東北人杨耙。 一個月前我還...
    沈念sama閱讀 46,286評論 2 360
  • 正文 我出身青樓赤套,卻偏偏與公主長得像,于是被迫代替她去往敵國和親珊膜。 傳聞我的和親對象是個殘疾皇子容握,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 43,465評論 2 348

推薦閱讀更多精彩內(nèi)容

  • 概 述 上篇文章中實現(xiàn)了基于升序鏈表的定時器,此定時器存在著當(dāng)定時器數(shù)量變多辅搬,效率也會變低的問題唯沮。對此,我們使用時...
    荏苒何從cc閱讀 1,208評論 0 1
  • 1.什么是定時器 定時器堪遂,顧名思義就是用來定時的,通俗的可以將其看作一個鬧鐘萌庆。在Linux應(yīng)用編程層上溶褪,當(dāng)需要實現(xiàn)...
    JalynFang閱讀 4,964評論 0 3
  • 在上一節(jié)已經(jīng)了解到Linux上使用各種定時器的優(yōu)缺點,接下來主要介紹Posix定時器践险。傳送門:關(guān)于Linux應(yīng)用層...
    JalynFang閱讀 9,443評論 0 4
  • 今天感恩節(jié)哎猿妈,感謝一直在我身邊的親朋好友。感恩相遇巍虫!感恩不離不棄彭则。 中午開了第一次的黨會,身份的轉(zhuǎn)變要...
    迷月閃星情閱讀 10,559評論 0 11
  • 彩排完占遥,天已黑
    劉凱書法閱讀 4,197評論 1 3