精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

Redis 如何高效實現定時任務

網絡 Redis
Redis通過單線程結合非阻塞事件輪詢機制實現高效的網絡IO和時間事件處理,這篇文章我們將從源碼的角度深入分析一下redis時間事件的設計與實現。

Redis通過單線程結合非阻塞事件輪詢機制實現高效的網絡IO和時間事件處理,這篇文章我們將從源碼的角度深入分析一下redis時間事件的設計與實現。

詳解redis中的時間事件

時間事件的定義

時間事件可以是單次到期執行銷毀,也可以是定時任務,對此redis對于時間事件統一封裝為aeTimeEvent對象,通過id來唯一標識一個事件,結合when_sec和when_ms記錄任務到期執行的秒和分,而執行時間事件的函數也是交由timeProc指針所指向的函數執行。 我們以一個redis定時執行的任務為例,如下所示,該結果通過when_sec和when_ms記錄秒之前的時間和毫秒的時間,一旦這個時間到了就會執行timeProc這個函數指針所指向的方法serverCron,該函數會定期執行各種任務,這一點筆者會在后文展開:

對應的我們給出時間事件的代碼描述,即位于ae.h這個頭文件中的aeTimeEvent 結構體,這就是對時間事件的封裝結構體,可以看到它除了筆者上述提到的核心字段以外,還有一個next指針用于連接下一個注冊的時間事件:

//時間事件
typedef struct aeTimeEvent {
    //時間事件的id全局遞增
    long long id; /* time event identifier. */
    long when_sec; /* seconds */
    //時間到達的時間
    long when_ms; /* milliseconds */
    //對應時間時間的處理器
    aeTimeProc *timeProc;
    //......
    //連接下一個時間時間
    struct aeTimeEvent *next;
} aeTimeEvent;

上文提到redis的時間事件是以鏈表的形式關聯起來,這里我們也給出時間時間統一管理對象,即時間輪詢器aeEventLoop ,它通過timeEventHead記錄第一個時間時間而后續的時間時間統一用時間時間的next指針進行管理:

對應我們也給出這段時間代碼的定義,即位于ae.h中aeEventLoop 的定義:

typedef struct aeEventLoop {
   //......
    //管理時間事件的列表
    aeTimeEvent *timeEventHead;
   //......
} aeEventLoop;

注冊時間事件

redis在服務器初始化階段,會注冊一個定時的時間事件,大約每1毫秒觸發一次,該事件主要做的是:

  • 更新redis全局時鐘,該時鐘用于全局變量獲取時間用的。
  • 隨機抽取redis內存數據庫中的樣本刪除過期的鍵值對。
  • 如果檢查到aof重寫完成,則進行刷盤操作。
  • 如果發現當前aof大小過大,則fork子進程進行aof重寫操作。
  • ......。

對應我們給出時間事件注冊的源碼段,即redis初始化時調用的方法initServer中的aeCreateTimeEvent,可以看到它將定時任務封裝為時間事件timeEvent,并設置時間間隔為1毫秒一次:

void initServer(void) {
     //......

    /* Create the serverCron() time event, that's our main way to process
     * background operations. */
    //創建時間事件注冊到eventLoop->timeEventHead中
    if(aeCreateTimeEvent(server.el, 1, serverCron, NULL, NULL) == AE_ERR) {
        redisPanic("Can't create the serverCron time event.");
        exit(1);
    }
 //......
}

輪詢處理時間事件

redis每次處理完所有用戶的請求之后,都會調用一次時間時間處理函數processTimeEvents,輪詢并處理就緒的時間事件,由此保證盡可能準時執行時間事件,如果事件時間非定時任務則執行完成直接刪除,反之設置下一次執行時間。這些步驟全部完成之后,返回本次處理的時間事件數:

我們給出處理時間循環的入口aeMain,可以看到該函數就是redis核心函數所在,它會循環調用aeProcessEvents處理各種事件:

void aeMain(aeEventLoop *eventLoop) {
    eventLoop->stop = 0;
    while (!eventLoop->stop) {
        if (eventLoop->beforesleep != NULL)
            eventLoop->beforesleep(eventLoop);
        //處理各種事件    
        aeProcessEvents(eventLoop, AE_ALL_EVENTS);
    }
}

不如aeProcessEvents可以看到該函數執行完所有用戶請求之后調用processTimeEvents方法獲取并執行就緒的時間事件:

int aeProcessEvents(aeEventLoop *eventLoop, int flags)
{
    //......

  //處理就緒的客戶端事件
        numevents = aeApiPoll(eventLoop, tvp);
        for (j = 0; j < numevents; j++) {
            aeFileEvent *fe = &eventLoop->events[eventLoop->fired[j].fd];
            int mask = eventLoop->fired[j].mask;
            int fd = eventLoop->fired[j].fd;
            int rfired = 0;

     /* note the fe->mask & mask & ... code: maybe an already processed
             * event removed an element that fired and we still didn't
             * processed, so we check if the event is still valid. */
            if (fe->mask & mask & AE_READABLE) {
                rfired = 1;
                fe->rfileProc(eventLoop,fd,fe->clientData,mask);
            }
            if (fe->mask & mask & AE_WRITABLE) {
                if (!rfired || fe->wfileProc != fe->rfileProc)
                    fe->wfileProc(eventLoop,fd,fe->clientData,mask);
            }
            processed++;
        }
    }
   
    //上述核心網絡IO事件完成后處理時間事件
    if (flags & AE_TIME_EVENTS)
        processed += processTimeEvents(eventLoop);

    return processed; /* return the number of processed file/time events */
}

最后我們就可以看到處理時間事件的核心代碼段,其內部會從timeEventHead開始輪詢就緒的時間事件,比對當前時間是否大于或者等于到期時間,如果是則執行當前時間事件,再判斷這個事件是否是定時事件,如果是則更新下次執行時間,反之刪除,最后累加本次處理的時間時間數:

static int processTimeEvents(aeEventLoop *eventLoop) {
    int processed = 0;
    aeTimeEvent *te;
    long long maxId;
    time_t now = time(NULL);

   //......
    if (now < eventLoop->lastTime) {
     //從時間事件頭開始
        te = eventLoop->timeEventHead;
        while(te) {
            te->when_sec = 0;
            te = te->next;
        }
    }
    eventLoop->lastTime = now;

    te = eventLoop->timeEventHead;
    maxId = eventLoop->timeEventNextId-1;
    //循環處理到期的時間事件
    while(te) {
        long now_sec, now_ms;
        long long id;

        if (te->id > maxId) {
            te = te->next;
            continue;
        }
        aeGetTime(&now_sec, &now_ms);
        //如果現在的事件大于到達時間
        if (now_sec > te->when_sec ||
            (now_sec == te->when_sec && now_ms >= te->when_ms))
        {
            int retval;

            id = te->id;
            //調用時間時間函數處理該事件
            retval = te->timeProc(eventLoop, id, te->clientData);
            //更新處理數
            processed++;
           //.....
            if (retval != AE_NOMORE) {//如果事件類型不是AE_NOMORE則說明是定時事件更新周期,反之刪除
                aeAddMillisecondsToNow(retval,&te->when_sec,&te->when_ms);
            } else {
                aeDeleteTimeEvent(eventLoop, id);
            }
            te = eventLoop->timeEventHead;
        } else {
            te = te->next;
        }
    }
    return processed;
}

redis對于時間事件實現上的優化

因為時間事件有些要求定期執行,所以redis為了保證時間執行的實時性,做了如下兩個優化:

  • 對于比較耗時的時間事件,例如AOF重寫,通過fork子進程異步完成:
  • 對于返回給客戶端套接字的內容,如果長度超過預設的值,會主動讓出線程執行權,避免時間時間饑餓。

對應的我們給出第一點時間時間對于aof重寫的核心代碼段,可以看到serverCron內部判斷如果當前沒有rdb和aof子進程,且需要進行aof重寫則調用rewriteAppendOnlyFileBackground函數fork子進程進行aof重寫:

int serverCron(struct aeEventLoop *eventLoop, long long id, void *clientData) {
  
   //......

    /* Start a scheduled AOF rewrite if this was requested by the user while
     * a BGSAVE was in progress. */
    //aof_rewrite_scheduled設置為1,且沒有其他持久化子進程則進行aof重寫,通過異步避免耗時
    if (server.rdb_child_pid == -1 && server.aof_child_pid == -1 &&
        server.aof_rewrite_scheduled)
    {
        rewriteAppendOnlyFileBackground();
    }

  //......
}

//fork子進程進行aof重寫
int rewriteAppendOnlyFileBackground(void) {
   //......
    if ((childpid = fork()) == 0) {//fork子進程進行aof重寫
        char tmpfile[256];

        /* Child */
        closeListeningSockets(0);
        redisSetProcTitle("redis-aof-rewrite");
        //生成一個tmp文件
        snprintf(tmpfile,256,"temp-rewriteaof-bg-%d.aof", (int) getpid());
        if (rewriteAppendOnlyFile(tmpfile) == REDIS_OK) {//重寫aof
            size_t private_dirty = zmalloc_get_private_dirty();

           //......
            exitFromChild(0);
        } else {
            exitFromChild(1);
        }
    } else {
        //......
    }
    return REDIS_OK; /* unreached */
}

而回復給客戶端結果的處理器sendReplyToClient內部也有一段,判斷如果寫入數totwritten 大于REDIS_MAX_WRITE_PER_EVENT (宏定義為64M),則直接中止寫入,break退出等到下一次循環處理,避免因為這個處理導致其他時間事件饑餓而導致事件執行延期:

void sendReplyToClient(aeEventLoop *el, int fd, void *privdata, int mask) {
    //......

    while(c->bufpos > 0 || listLength(c->reply)) {
     //......
        //對于文件事件數據寫入超長會讓出執行權讓時間事件能夠盡可能的執行
        server.stat_net_output_bytes += totwritten;
        if (totwritten > REDIS_MAX_WRITE_PER_EVENT &&
            (server.maxmemory == 0 ||
             zmalloc_used_memory() < server.maxmemory)) break;
    }
     //......
}
責任編輯:趙寧寧 來源: 寫代碼的SharkChili
相關推薦

2023-11-07 07:47:35

Topic線程PUSH

2017-03-13 09:12:00

TCP數據結構請求包

2020-12-21 07:31:23

實現單機JDK

2024-12-27 08:24:55

2024-02-28 09:54:07

線程池配置

2023-12-19 08:09:06

Python定時任務Cron表達式

2024-09-09 08:11:12

2024-05-13 09:49:30

.NETQuartz庫Cron表達式

2023-11-16 09:30:27

系統任務

2021-11-22 12:35:40

Python命令定時任務

2024-02-26 11:12:33

定時任務線程

2024-01-31 08:38:57

Python定時任務函數

2024-01-22 08:53:00

策略任務RocketMQ

2024-05-31 13:07:29

.NET Core定時任務編程

2023-08-08 08:35:28

web框架Hosting模塊

2010-03-10 15:47:58

crontab定時任務

2009-10-28 10:05:29

Ubuntucrontab定時任務

2012-02-07 13:31:14

SpringJava

2024-10-15 16:41:35

2020-08-05 07:37:29

任務系統定時
點贊
收藏

51CTO技術棧公眾號

欧美自拍偷拍网| 国产人妻777人伦精品hd| 亚洲永久精品视频| 久久国产电影| 日韩三级视频在线观看| 男女猛烈激情xx00免费视频| 日韩在线免费看| 久久亚洲美女| 久久久999精品| www国产视频| 999国产精品亚洲77777| 亚洲精品视频免费看| 久久精品国产精品青草色艺| 中文字幕一区二区三区四区视频 | 国产精品伊人色| 66m—66摸成人免费视频| 天天干天天舔天天操| 日韩精品久久久久久久软件91| 精品久久久久久久久久ntr影视| 色一情一区二区三区四区| www.亚洲天堂.com| 日韩精品欧美精品| 欧美精品video| 欧美色图17p| 色老板在线视频一区二区| 欧美日本视频在线| 大j8黑人w巨大888a片| 免费超碰在线| 久久久影视传媒| 99中文字幕| 中文字幕第三页| 亚洲永久字幕| 久久久综合av| 日本黄色小说视频| 日韩一区二区在线| 亚洲欧美成人在线| 无码国产精品一区二区免费式直播 | 成人免费在线播放视频| 你懂的视频在线一区二区| 国产熟女一区二区丰满| 亚洲欧洲另类| 九九热精品视频在线播放| 综合 欧美 亚洲日本| 一本色道久久综合狠狠躁的番外| 日韩免费观看高清完整版在线观看| 99视频在线免费| 亚洲国产福利| 欧美日韩国产一区在线| 国产中文字幕乱人伦在线观看| 麻豆传媒在线观看| 久久久久久久精| 快播亚洲色图| 肉丝一区二区| 久久综合色天天久久综合图片| 国产精品一区二区av| 精品久久久免费视频| 国产美女娇喘av呻吟久久| 成人久久久久久久| 91成品人影院| 韩国理伦片一区二区三区在线播放| 国产精品视频区| 欧美日韩a v| 日韩av一区二区三区四区| 欧美在线视频网站| 国产成人精品777777| 先锋亚洲精品| 日本欧美爱爱爱| 国产suv精品一区二区33| 性8sex亚洲区入口| 国产aⅴ夜夜欢一区二区三区 | 欧美视频在线一区二区三区| 玩弄japan白嫩少妇hd| 蜜桃视频成人m3u8| 欧美日韩情趣电影| 99九九精品视频| 一区三区自拍| 日韩精品极品视频| 亚洲天堂视频一区| 成人三级视频| 欧美另类xxx| 日本三级午夜理伦三级三| 夜夜爽99久久国产综合精品女不卡 | 黑人糟蹋人妻hd中文字幕| 欧美黑人粗大| 91精品国产一区二区三区蜜臀 | 影音先锋黄色网址| 国产裸体歌舞团一区二区| 国产欧美日韩在线播放| 久草视频视频在线播放| 国产精品久久影院| 日韩久久久久久久久久久久| 黑人精品一区| 在线成人av影院| 日批在线观看视频| 日韩成人影院| 午夜精品免费视频| 在线视频你懂得| 成人综合在线网站| 日本一区二区三区视频在线观看| 蜜桃视频网站在线| 亚洲国产精品人人做人人爽| 欧美黑人又粗又大又爽免费| 99综合久久| 在线电影一区二区三区| 性鲍视频在线观看| 亚洲资源网站| 色偷偷9999www| 国产人妻精品一区二区三区不卡| 黄色亚洲免费| 国产精品久久久久久久久久尿| 一区二区三区播放| 不卡的电视剧免费网站有什么| 久久人人97超碰人人澡爱香蕉| 精品久久久久一区二区三区| 国产香蕉久久精品综合网| 4444在线观看| 久久久久久久| 欧美片在线播放| 波多野结衣一本| 亚洲男女av一区二区| 97视频在线观看免费| 亚洲自拍偷拍另类| 99久久精品免费看国产免费软件| 日韩av不卡在线播放| 欧美人动性xxxxz0oz| 欧美色老头old∨ideo| 91丝袜在线观看| 91精品亚洲| 欧美伊久线香蕉线新在线| 国产精品视频a| 人人九九精品视频| 久久综合九色综合97婷婷 | 日韩在线影视| 久久亚洲综合国产精品99麻豆精品福利 | 国产一区二区三区免费观看在线| 日韩免费视频一区二区| 欧美黄色激情视频| 激情亚洲网站| 亚洲a级在线播放观看| 青青操视频在线| 亚洲精品少妇30p| 午夜xxxxx| 成人国产精品一级毛片视频| 久久久久久久久国产精品| 一区二区三区免费在线| 久久久久久黄色| 欧美二区在线视频| 自拍偷拍亚洲| 久久亚洲电影天堂| 91麻豆精品在线| wwwwxxxxx欧美| 分分操这里只有精品| 精品国产亚洲一区二区三区| 一区三区二区视频| 国产黄色免费观看| 99精品视频在线免费观看| 国产一区二区三区在线免费| 国产精品亚洲综合在线观看| 色吧影院999| 国产性生活视频| 99国产精品久久久久| 国产婷婷一区二区三区| 波多野结衣一区二区三区免费视频| 日韩最新中文字幕电影免费看| 极品国产91在线网站| 91美女片黄在线观看| 欧美日韩一道本| 综合亚洲自拍| 欧美怡红院视频一区二区三区| 五月婷婷六月丁香| 午夜精品国产更新| 欧美日韩人妻精品一区在线| 好吊日精品视频| 91精品国产高清久久久久久91裸体| 污视频在线免费观看网站| 91麻豆精品国产91久久久久久 | 国产无遮挡裸体免费久久| 欧美理论电影在线播放| 欧美一区二区三区激情| 五月婷婷久久综合| 欧美 日韩 国产 成人 在线观看 | 日韩在线视频导航| 中文字幕在线视频第一页| 日本一区二区三区久久久久久久久不| 久久这里只精品| 国产精品久久久久久影院8一贰佰| 成人网在线视频| 色呦呦在线看| 日韩成人在线电影网| 日韩欧美在线观看免费| 中文字幕不卡在线| 女同激情久久av久久| 亚洲深夜av| 欧美日韩精品综合| 成人精品一区二区三区电影| 久久视频精品在线| 亚洲欧美另类一区| 亚洲成av人片一区二区| 国产一区二区三区四区在线| 极品美女销魂一区二区三区免费| 国产性生活免费视频| 任你弄精品视频免费观看| 国产精品96久久久久久又黄又硬| 69av在线| 亚洲精品国产免费| 无码视频在线观看| 亚洲免费在线看| 国产精品久久久免费观看| 蜜臀久久99精品久久久久宅男| 成年人三级视频| 老汉色老汉首页av亚洲| 国产精品成av人在线视午夜片| 国产一区久久精品| 亚洲级视频在线观看免费1级| 中文字幕在线日本| 亚洲制服丝袜av| 国产三级精品三级观看| www.99精品| 天天操狠狠操夜夜操| 99国产精品视频免费观看一公开| 日韩欧美亚洲精品| 欧美一区=区三区| 欧洲亚洲免费视频| 二区三区在线观看| 亚洲欧美国产制服动漫| 国内精品久久久久久久久久久 | 日韩伦理在线免费观看| 日韩欧美午夜| 欧美在线一二三区| 18国产精品| 国产噜噜噜噜久久久久久久久| 国产精品偷拍| www.日韩系列| 天堂a√在线| 日韩午夜激情视频| 一区二区视频播放| 欧美性猛交xxxx| 免费又黄又爽又色的视频| 日本一区二区视频在线| 天堂www中文在线资源| 国产一区二区中文字幕| 日本www高清视频| 99在线观看免费视频精品观看| 国产三级中文字幕| 成人羞羞网站入口免费| 日韩av电影免费播放| 希岛爱理av免费一区二区| 日韩成人在线网站| 成年人网站av| 日韩精品欧美精品| 国产一级不卡毛片| 久久性天堂网| 免费日韩视频在线观看| 国产欧美一级| 午夜肉伦伦影院| 国产一区久久| 女人色极品影院| 欧美福利影院| 影音先锋成人资源网站| 国产精品精品| 亚洲国产日韩美| 亚洲国产精品成人| 日本福利视频导航| 五月天久久网站| 欧美一级精品在线| 成人黄色免费网址| 欧美激情中文不卡| 污污视频网站在线免费观看| 欧美韩国日本综合| 国产欧美一区二区三区在线观看视频| 久久综合丝袜日本网| 国产精品成人在线视频| 中文在线免费一区三区高中清不卡| 性少妇bbw张开| 久久精品亚洲乱码伦伦中文| 91传媒理伦片在线观看| 久久久久久综合| 男人天堂av电影| 亚洲国产高清在线| 亚洲熟女少妇一区二区| 国产精品女同一区二区三区| 国产欧美一区二区三区不卡高清| 久久久人成影片一区二区三区在哪下载| 欧洲精品在线视频| 成人黄色免费观看| 91亚洲一区精品| 好吊妞国产欧美日韩免费观看网站| 国产一区精品在线| 亚洲免费成人av在线| 亚洲国产日韩欧美| 欧美三级乱码| 成人一区二区三| 黄网站免费久久| 亚洲图片欧美另类| 久久这里只有精品视频网| 免费黄色在线视频| 亚洲国产高清不卡| 久草视频免费在线| 欧美性猛交xxxx黑人| 91免费精品国偷自产在线| 成人免费视频观看| 国产在线欧美日韩| 欧美日韩激情在线一区二区三区| 亚洲国产另类久久久精品极度| 亚洲免费二区| 精品中文字幕av| 日韩极品在线观看| www.久久com| 成人免费福利片| 国产jizz18女人高潮| 亚洲国产综合色| 这里只有精品国产| 亚洲精品一区二区三区福利| 最新97超碰在线| 午夜精品www| 国产传媒免费观看| 精品一区二区免费在线观看| 极品白嫩的小少妇| 国产亚洲欧美在线| 日韩三级视频在线| 777色狠狠一区二区三区| 性xxxx视频播放免费| 日韩视频免费在线观看| 欧美亚洲韩国| 国产精品美女久久久久av福利| 欧美在线观看视频一区| www插插插无码视频网站| 精品一区二区三区免费播放| 国产特黄级aaaaa片免| 一区二区不卡在线视频 午夜欧美不卡在 | 九色91porny| 久久精品亚洲麻豆av一区二区 | 无人码人妻一区二区三区免费| 91丨porny丨在线| 黄色一级视频在线观看| 欧美三级在线看| 午夜影院免费体验区| 欧美成人亚洲成人| 久久精品资源| 精品国产乱码久久久久久蜜柚| 久久精品欧美一区| 手机视频在线观看| 日韩伦理三区| 91超碰在线免费观看| 日本a口亚洲| 天天干天天操天天做| 久久久午夜精品| 欧美三日本三级少妇99| 日韩视频一区二区三区 | 久久成人国产精品| 激情亚洲小说| 婷婷四月色综合| 天堂久久一区二区三区| 欧美激情亚洲色图| 日韩欧美中文第一页| 亚洲av电影一区| 国内外成人免费激情在线视频网站 | 2021国产精品视频| 精品成人自拍视频| 久久天天东北熟女毛茸茸| 国产精品一区二区在线播放| 情侣偷拍对白清晰饥渴难耐| 欧美日韩色综合| 一广人看www在线观看免费视频| 国产日韩视频在线观看| 欧美日韩国产一区二区三区不卡| 熟妇人妻va精品中文字幕| 久久久噜噜噜久久中文字幕色伊伊| 国产成人无码精品久在线观看| 精品国产乱码久久久久久1区2区| 日韩123区| 精品91免费| 一本久道久久综合狠狠爱| 国产高潮视频在线观看| 亚洲欧美国产高清| 国产福利免费视频| 欧美激情日韩图片| 国产女人18毛片水真多18精品| 无罩大乳的熟妇正在播放| 中文字幕av资源一区| 国产在成人精品线拍偷自揄拍| 久久伊人精品天天| 成人av激情人伦小说| 国产一区亚洲二区三区| 国产精品网曝门| 国产人妻精品一区二区三区| 欧美激情视频一区二区三区不卡| 偷拍亚洲精品| 超碰在线播放91| 樱桃国产成人精品视频| 国产日韩在线观看一区| 2018日韩中文字幕| 精品一级毛片| 97超碰免费在线观看| 欧美日韩亚洲一区二| 91精品大全| http;//www.99re视频| 国产精品毛片在线| 神马久久精品综合| 亚洲第一区中文99精品|