NGINX 定時器

寫在前面

寫NGINX系列的隨筆,一來總結學到的東西,二來記錄下疑惑的地方,在接下來的學習過程當中去解決疑惑。linux

也但願一樣對NGINX感興趣的朋友可以解答個人疑惑,或者共同探討研究。nginx

整個NGINX系列的文章中,我會將個人疑惑用紅色標出,但願能遇到前輩在評論中給我解答迷津。函數

定時器

在介紹定時器以前,先簡要說下nginx處理事件的流程和方式。post

Worker進程的主要流程:性能

 

 1 static void
 2 ngx_worker_process_cycle(ngx_cycle_t *cycle, void *data){
 3   for(;;) {
 4   if(ngx_exiting) {}
 5   ngx_process_events_and_timers(cycle);
 6   if (ngx_terminate) {}
 7   if (ngx_quit) {}
 8   if (ngx_reopen) {}
 9 }
10 
11 void
12 ngx_process_events_and_timers(ngx_cycle_t *cycle) {
13   (void) ngx_process_events(cycle, timer, flags);
14 }

 

ngx_process_events調用epoll(linux下)實現事件的處理。學習

ngx_process_events處理事件有兩種方式:一是直接調用處理函數處理,二是將事件放到post隊列中,函數返回後再處理隊列中的事件。ui

在使用了 NGX_POST_EVENTS標記時,ngx_process_events不直接處理事件,將事件放到Post隊列中,待函數返回,再在隊列中取出事件處理。spa

由於調用ngx_process_events會加鎖(爲何加鎖?),函數返回後,將鎖釋放再處理事件,能夠減小鎖的佔用時間。code

 

以上是nginx處理事件的大致方式,下面介紹nginx中定時器的實現。blog

Nginx定時器使用紅黑樹組織(爲何使用紅黑樹,紅黑樹效率高到哪?能夠研究下)存儲,這個很少說。

Nginx定時器的觸發有兩種方式,第一種是設置時間信號。

ngx_event_process_init函數中 ,設置了時間信號,每隔固定時間觸發,時間信號的處理函數,只是設置ngx_event_timer_alarm = 1,但他會中斷ngx_process_eventsepoll_wait的處理,epoll_wait返回後,調用ngx_time_update更新時間,接着返回到函數ngx_process_events_and_timers中處理,ngx_process_events_and_timers中,會調用ngx_event_expire_timers,查詢超時的事件並處理。

但這種方式有個問題,若是事件信號是在處理IO事件時(epoll_wait調用以後)發生的,那麼定時器的查詢遍歷,只能到下一次epoll_wait調用時纔會處理,若是這時有IO事件發生,那麼epoll_wait能夠當即返回,而後由於上次信號發生已經置ngx_event_timer_alarm = 1,能夠當即更新時間,ngx_process_events返回後能夠處理定時器事件。但若是沒有IO事件發生,epoll_wait會阻塞到下次時間信號到來,而後處理定時器事件,這樣豈不大大下降了定時器的精確度。這塊nginx怎麼處理的?

 

另外每隔固定時間(具體設置的時間信號的時間)才更新時間值,甚至多是兩倍時間信號的時間才更新時間值,那麼代碼中在插入的定時器,實際觸發時間和理論時間就會有這麼大的偏差。是否是這樣呢?

 

Nginx定時器的第二種觸發方式是利用epoll_wait的超時。

每次在調用epoll_wait以前,nginx都會取得下一個最小(最先要觸發)的定時器的時間值,而後拿這個值做爲epoll_wait的超時時間。這樣epoll_wait在返回後就能夠處理超時事件了。既能夠在頻繁IO的狀況下處理超時,又能夠在IO少許的狀況下處理超時。

這種方式epoll_wait返回後,都會先更新時間,這樣epoll_wait返回後,在IO事件的處理代碼中加入定時器,偏差不會太大,由於時間剛剛被更新。

但這個方法的問題是,IO頻繁的狀況下,也會頻繁更新時間,是否會影響性能?

這兩種方式各自的優缺點是哪些呢?

相關文章
相關標籤/搜索