日日操夜夜添-日日操影院-日日草夜夜操-日日干干-精品一区二区三区波多野结衣-精品一区二区三区高清免费不卡

公告:魔扣目錄網為廣大站長提供免費收錄網站服務,提交前請做好本站友鏈:【 網站目錄:http://www.ylptlb.cn 】, 免友鏈快審服務(50元/站),

點擊這里在線咨詢客服
新站提交
  • 網站:51998
  • 待審:31
  • 小程序:12
  • 文章:1030137
  • 會員:747

Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

Tomcat的啟動核心流程

  前面給大家介紹了Tomcat中的生命周期的設計,掌握了這塊對于我們分析Tomcat的核心流程是非常有幫助的,也就是我們需要創建相關的核心組件,比如Server,Service肯定都繞不開生命周期的方法。

Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

1.啟動的入口

  你可以通過腳本來啟動Tomcat服務(startup.bat),但如果你看過腳本的命令,你會發現最終調用的還是Bootstrap中的main方法,所以我們需要從main方法來開始

Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

  然后我們去看main方法中的代碼,我們需要重點關注的方法有三個

  1. bootstrap.init()方法
  2. load()方法
  3. start()方法

  也就是在這三個方法中會完成Tomcat的核心操作。

2.init方法

  我們來看下init方法中的代碼,非核心的我們直接去掉

    public void init() throws Exception {
        // 創建相關的類加載器
        initClassLoaders();
        // 省略部分代碼...
        // 通過反射創建了 Catalina 類對象
        Class<?> startupClass = catalinaLoader
            .loadClass("org.Apache.catalina.startup.Catalina");
        // 創建了 Catalina 實例
        Object startupInstance = startupClass.getConstructor().newInstance();

        // 省略部分代碼...
        String methodName = "setParentClassLoader";
        Class<?> paramTypes[] = new Class[1];
        paramTypes[0] = Class.forName("JAVA.lang.ClassLoader");
        Object paramValues[] = new Object[1];
        paramValues[0] = sharedLoader;
        // 把 sharedLoader 設置為了 commonLoader的父加載器
        Method method =
            startupInstance.getClass().getMethod(methodName, paramTypes);
        method.invoke(startupInstance, paramValues);

        // Catalina 實例 賦值給了 catalinaDaemon
        catalinaDaemon = startupInstance;
    }
  1. 首先是調用了initClassLoaders()方法,這個方法會完成對應的ClassLoader的創建,這個比較重要,后面專門寫一篇文章來介紹。
  2. 通過反射的方式創建了Catalina的類對象,并通過反射創建了Catalina的實例
  3. 設置了類加載器的父子關系
  4. 用過成員變量catalinaDaemon記錄了我們創建的Catalina實例

  這個是通過bootstrap.init()方法我們可以獲取到的有用的信息。然后我們繼續往下面看。

3.load方法

  然后我們來看下load方法做了什么事情,代碼如下:

    private void load(String[] arguments) throws Exception {

        // Call the load() method
        String methodName = "load"; // load方法的名稱
        Object param[];
        Class<?> paramTypes[];
        if (arguments==null || arguments.length==0) {
            paramTypes = null;
            param = null;
        } else {
            paramTypes = new Class[1];
            paramTypes[0] = arguments.getClass();
            param = new Object[1];
            param[0] = arguments;
        }
        // catalinaDaemon 就是在 init中創建的 Catalina 對象
        Method method =
            catalinaDaemon.getClass().getMethod(methodName, paramTypes);
        if (log.isDebugEnabled()) {
            log.debug("Calling startup class " + method);
        }
        // 會執行 Catalina的load方法
        method.invoke(catalinaDaemon, param);
    }

  上面的代碼非常簡單,通過注釋我們也可以看出該方法的作用是調用 Catalina的load方法。所以我們還需要加入到Catalina的load方法中來查看,代碼同樣比較長,只留下關鍵代碼

    public void load() {

        if (loaded) {
            return; // 只能被加載一次
        }
        loaded = true;

        initDirs(); // 廢棄的方法

        // Before digester - it may be needed
        initNaming(); // 和JNDI 相關的內容 忽略

        // Create and execute our Digester
        // 創建并且執行我們的 Digester 對象  Server.xml
        Digester digester = createStartDigester();

        // 省略掉了 Digester文件處理的代碼

        getServer().setCatalina(this); // Server對象綁定 Catalina對象
        getServer().setCatalinaHome(Bootstrap.getCatalinaHomeFile());
        getServer().setCatalinaBase(Bootstrap.getCatalinaBaseFile());

        // Stream redirection
        initStreams();
        // 省略掉了部分代碼...
         getServer().init(); // 完成 Server  Service Engine Connector等組件的init操作

    }

把上面的代碼簡化后我們發現這個Load方法其實也是蠻簡單的,就做了兩件事。

  1. 通過Apache下的Digester組件完成了Server.xml文件的解析
  2. 通過getServer().init() 方法完成了Server,Service,Engin,Connector等核心組件的初始化操作,這塊和前面的LifecycleBase呼應起來了。
Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

  如果生命周期的內容不清楚,請看上一篇文章的介紹。

4.start方法

  最后我們來看下start方法的代碼。

    public void start() throws Exception {
        if (catalinaDaemon == null) {
            init(); // 如果 catalinaDaemon 為空 初始化操作
        }
        // 獲取的是 Catalina 中的 start方法
        Method method = catalinaDaemon.getClass().getMethod("start", (Class [])null);
        // 執行 Catalina 的start方法
        method.invoke(catalinaDaemon, (Object [])null);
    }

  上面的代碼邏輯也很清楚,就是通過反射的方式調用了Catalina對象的start方法。所以進入Catalina的start方法中查看。

    public void start() {

        if (getServer() == null) {
            load(); // 如果Server 為空 重新 init 相關的組件
        }

        if (getServer() == null) {
            log.fatal("Cannot start server. Server instance is not configured.");
            return;
        }

        // Start the new server  關鍵方法--->啟動Server
        try {
            getServer().start();
        } catch (LifecycleException e) {
            // 省略...
        }

        // 省略...

        // Register shutdown hook  注冊關閉的鉤子
        if (useShutdownHook) {
            // 省略...
        }

        if (await) {
            await();
            stop();
        }
    }

  通過上面的代碼我們可以發現核心的代碼還是getServer.start()方法,也就是通過Server對象來嵌套的調用相關注解的start方法。

Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

5.核心流程的總結

我們可以通過下圖來總結下Tomcat啟動的核心流程

Tomcat源碼篇-簡單到超乎想象的啟動流程

 

image.png

  從圖中我們可以看到Bootstrap其實沒有做什么核心的事情,主要還是Catalina來完成的。

  本文只是從Tomcat的啟動核心流程來講解,還有些具體的實現細節沒有介紹到,會在后續的文章中給大家細細到來,歡迎關注收藏哦!

分享到:
標簽:Tomcat
用戶無頭像

網友整理

注冊時間:

網站:5 個   小程序:0 個  文章:12 篇

  • 51998

    網站

  • 12

    小程序

  • 1030137

    文章

  • 747

    會員

趕快注冊賬號,推廣您的網站吧!
最新入駐小程序

數獨大挑戰2018-06-03

數獨一種數學游戲,玩家需要根據9

答題星2018-06-03

您可以通過答題星輕松地創建試卷

全階人生考試2018-06-03

各種考試題,題庫,初中,高中,大學四六

運動步數有氧達人2018-06-03

記錄運動步數,積累氧氣值。還可偷

每日養生app2018-06-03

每日養生,天天健康

體育訓練成績評定2018-06-03

通用課目體育訓練成績評定