Tomcat-LifeCycle

概述

Tomcat-整體架構(gòu)中看到Tomcat涉及眾多有生命周期概念的組件,包括Server/Service/Engine/Host/Context/Wrapper,那么如何統(tǒng)一管理這些組件(也即如何管理Tomcat生命周期)就是設(shè)計的要點,本節(jié)就來看下Tomcat的設(shè)計。

LifeCycle.png

1. 組合模式

把生命周期的過程抽象成LifeCycle接口,每個具體的組件去實現(xiàn)這些方法

public interface Lifecycle {
    public void init() throws LifecycleException;
    public void start() throws LifecycleException;
    public void stop() throws LifecycleException;
    public void destroy() throws LifecycleException;
    .......
}

在父組件的init()方法里需要創(chuàng)建子組件并調(diào)用子組件的init()方法。同樣,在父組件的start()方法里也需要調(diào)用子組件的start()方法,因此調(diào)用者可以無差別的調(diào)用各組件的init()方法和start()方法,這就是組合模式的使用,并且只要調(diào)用最頂層組件,也就是Server組件的init()和start()方法,整個Tomcat就被啟動起來了。

2. 觀察者模式

在LifeCycle接口里加入了管理監(jiān)聽器的方法,另外定義一個Enum來表示組件有哪些狀態(tài),以及處在什么狀態(tài)會觸發(fā)什么樣的事件;比如NEW狀態(tài)表示組件剛剛被實例化;而當init()方法被調(diào)用時,狀態(tài)就變成INITIALIZING狀態(tài),這個時候,就會觸發(fā)BEFORE_INIT_EVENT事件,如果有監(jiān)聽器在監(jiān)聽這個事件,它的方法就會被調(diào)用。

public interface Lifecycle {
    public void addLifecycleListener(LifecycleListener listener);
    public LifecycleListener[] findLifecycleListeners();
    public void removeLifecycleListener(LifecycleListener listener);
    public LifecycleState getState();
    public String getStateName();
    ......
}
public enum LifecycleState {
    NEW(false, null),
    INITIALIZING(false, Lifecycle.BEFORE_INIT_EVENT),
    INITIALIZED(false, Lifecycle.AFTER_INIT_EVENT),
    STARTING_PREP(false, Lifecycle.BEFORE_START_EVENT),
    STARTING(true, Lifecycle.START_EVENT),
    STARTED(true, Lifecycle.AFTER_START_EVENT),
    STOPPING_PREP(true, Lifecycle.BEFORE_STOP_EVENT),
    STOPPING(false, Lifecycle.STOP_EVENT),
    STOPPED(false, Lifecycle.AFTER_STOP_EVENT),
    DESTROYING(false, Lifecycle.BEFORE_DESTROY_EVENT),
    DESTROYED(false, Lifecycle.AFTER_DESTROY_EVENT),
    FAILED(false, null);
    .....
}
 // 狀態(tài)流轉(zhuǎn)過程
 *            start()
 *  -----------------------------
 *  |                           |
 *  | init()                    |
 * NEW -?-- INITIALIZING        |
 * | |           |              |     ------------------?-----------------------
 * | |           |auto          |     |                                        |
 * | |          \|/    start() \|/   \|/     auto          auto         stop() |
 * | |      INITIALIZED --?-- STARTING_PREP --?- STARTING --?- STARTED --?---  |
 * | |         |                                                            |  |
 * | |destroy()|                                                            |  |
 * | --?-----?--    ------------------------?--------------------------------  ^
 * |     |          |                                                          |
 * |     |         \|/          auto                 auto              start() |
 * |     |     STOPPING_PREP ----?---- STOPPING ------?----- STOPPED -----?-----
 * |    \|/                               ^                     |  ^
 * |     |               stop()           |                     |  |
 * |     |       --------------------------                     |  |
 * |     |       |                                              |  |
 * |     |       |    destroy()                       destroy() |  |
 * |     |    FAILED ----?------ DESTROYING ---?-----------------  |
 * |     |                        ^     |                          |
 * |     |     destroy()          |     |auto                      |
 * |     --------?-----------------    \|/                         |
 * |                                 DESTROYED                     |
 * |                                                               |
 * |                            stop()                             |
 * ----?-----------------------------?------------------------------
 *

組件的init()和start()調(diào)用是由它的父組件的狀態(tài)變化觸發(fā)的,上層組件的初始化會觸發(fā)子組件的初始化,上層組件的啟動會觸發(fā)子組件的啟動,因此把組件的生命周期定義成一個個狀態(tài),把狀態(tài)的轉(zhuǎn)變看作是一個事件。而事件是有監(jiān)聽器的,在監(jiān)聽器里可以實現(xiàn)一些邏輯,并且監(jiān)聽器也可以方便的添加和刪除,這就是典型的觀察者模式。

3. 模板方法模式

Tomcat定義一個基類LifeCycleBase來實現(xiàn)LifeCycle接口,把一些公共的邏輯放到基類中去,比如生命狀態(tài)的轉(zhuǎn)變與維護、生命事件的觸發(fā)以及監(jiān)聽器的添加和刪除等,而子類就負責實現(xiàn)自己的初始化、啟動和停止等方法。

// 以init方法為例分析
public final synchronized void init() throws LifecycleException {
    if (!state.equals(LifecycleState.NEW)) {
        invalidTransition(Lifecycle.BEFORE_INIT_EVENT);
    }

    try {
        setStateInternal(LifecycleState.INITIALIZING, null, false);
        initInternal();
        setStateInternal(LifecycleState.INITIALIZED, null, false);
    } catch (Throwable t) {
        handleSubClassException(t, "lifecycleBase.initFail", toString());
    }
}

1. invalidTransition檢查狀態(tài)的合法性,比如當前狀態(tài)必須是NEW然后才能進行初始化。
2. setStateInternal觸發(fā)INITIALIZING事件的監(jiān)聽器:在這個setStateInternal方法里,會調(diào)用監(jiān)聽器的業(yè)務(wù)方法。
3. initInternal調(diào)用具體子類實現(xiàn)的抽象方法initInternal()方法。我在前面提到過,為了實現(xiàn)一鍵式啟動,具體組件在實現(xiàn)initInternal()方法時,又會調(diào)用它的子組件的init()方法。
4. setStateInternal子組件初始化后,觸發(fā)INITIALIZED事件的監(jiān)聽器,相應(yīng)監(jiān)聽器的業(yè)務(wù)方法就會被調(diào)用。

LifeCycleBase實現(xiàn)LifeCycle接口的方法并設(shè)置為final,在實現(xiàn)方法中實現(xiàn)公共邏輯,提供對應(yīng)的xxxInternal()方法供子類實現(xiàn),這是典型的模板方法模式。

-------over--------

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容