`
clusty
  • 浏览: 5967 次
  • 性别: Icon_minigender_1
  • 来自: 北京
最近访客 更多访客>>
文章分类
社区版块
存档分类
最新评论

<zt>Tomcat 原理解说:启动过程分析

阅读更多
从开始学Java,使用网页编程,我们的程序就一直在web容器内运行。容器的概念也是彷佛距离我们相当遥远。Spring给了我们一扇窗户,让我们理解轻量级容器的解决方案。非常庆幸我们所处在一个开源的Java世界,所有的优秀软件都可以在代码级进行亲密接触。
Tomcat结构写的相当之好,极易阅读。本系列文档计划着重以下3个方面:
1. Tomcat 启动过程分析。分析tomcat的启动过程
2. Tomcat Web容器功能分析。分析web容器的实现方法
3. Tomcat 其他配置说明。一些我们目前应用较少的Tomcat配置项说明
注意,本文档分析基于Tomcat5.028完成。
Tomcat的启动过程简单说起来,就是读取配置文件server.xml,然后对其进行实例化的过程。
1. 启动命令行参数
命令行方式下,直接启动startup.bat/.sh即可。bat文件目的就在于获得启动参数。一般情况下,使用类似如下所示的启动方法。注意:bootstrap.jar文件中,还指定了3个包在classpath中。
java -classpath "d:\tools\jdk1.5.0\lib\tools.jar;D:\Tomcat-5.0.28\bin\bootstrap.jar" -Dcatalina.base="D:\Tomcat-5.0.28" -Dcatalina.home="D:\Tomcat-5.0.28" -Djava.io.tmpdir="D:\Tomcat-5.0.28\temp" org.apache.catalina.startup.Bootstrap  start
Bootstrap是Tomcat的入口。比如启动、关闭、重启都是通过这个类实现对tomcat的控制。

2. ClassLoader的使用
Tomcat对不同的模块可能使用不同的ClassLoader加载。这也就是为什么很多类不在启动的 classpath中,却可以被它调用的原因。
下面是Bootstrap初始化ClassLoader的方法:
private void initClassLoaders() {
    try {
        ClassLoaderFactory.setDebug(debug);
        commonLoader = createClassLoader("common", null);
        catalinaLoader = createClassLoader("server", commonLoader);
        sharedLoader = createClassLoader("shared", commonLoader);
    } catch (Throwable t) {
        log("Class loader creation threw exception", t);
        System.exit(1);
    }
}

下图是Tomcat用户手册上看到的。
       Bootstrap
          |
       System
          |
       Common
      /      \
Catalina   Shared
(server)     /   \
      Webapp1  Webapp2 ... 
Bootstrap 是JVM提供的
System是在classpath中提供的
Common包含配置文件/org/apache/catalina /startup/catalina.properties中指定的类库支持
Catalina和Shared都从Common中继承,包含的类库也在上面配置文件中指定。
WebappX在部署单个Tomcat5实例时指定。一个webapp下面的类库对另外一个是不可见的
Tomcat 加载类的顺序和普通的不太一样,如下:
Bootstrap classes of your JVM
System class loader classses (described above)
/WEB-INF/classes of your web application
/WEB-INF/lib/*.jar of your web application
$CATALINA_HOME/common/classes
$CATALINA_HOME/common/endorsed/*.jar
$CATALINA_HOME/common/lib/*.jar
$CATALINA_BASE/shared/classes
$CATALINA_BASE/shared/lib/*.jar
注意,如果希望不使用JVM本身提供的类。这时可以使用jdk的endorsed 特性。

3. Catalina类的作用
如果要启动Tomcat,那么一个 org.apache.catalina.startup.Catalina实例就生成,由它完成接下来的工作。
下面是它的启动代码
    public void start() {
        if (server == null) {
            load();
        }
        long t1 = System.currentTimeMillis();
        // Start the new server
        if (server instanceof Lifecycle) {
            try {
                ((Lifecycle) server).start();
            } catch (LifecycleException e) {
                log.error("Catalina.start: ", e);
            }
        }
        long t2 = System.currentTimeMillis();
        log.info("Server startup in " + (t2 - t1) + " ms");
        try {
            // Register shutdown hook
            if (useShutdownHook) {
                if (shutdownHook == null) {
                    shutdownHook = new CatalinaShutdownHook();
                }
                Runtime.getRuntime().addShutdownHook(shutdownHook);
            }
        } catch (Throwable t) {
        }
        if (await) {
            await();
            stop();
        }
    }
启动过程先载入配置文件,然后根据配置文件启动的Server实例启动实例,在实例中注册关闭钩子。
接下来的工作就是等待发出关闭指令或重启指令了。
4. Server对象的生成
服务的生成就是根据配置文件server.xml,实例化的对象。对象实例化过程中,会做载入webapp,在特定端口等待客户连接等工作。
从server.xml到对象的映射是通过commons-digester.jar包完成的。这个包的一个主要功能就是映射 xml到java对象。
catalina类的方法createStartDigester完成了这个工作。部分代码如下
Digester digester = new CatalinaDigester();
digester.setValidating(false);
digester.setClassLoader(StandardServer.class.getClassLoader());

// Configure the actions we will be using
digester.addObjectCreate("Server","org.apache.catalina.core.StandardServer", "className");
digester.addSetProperties("Server");
digester.addSetNext("Server","setServer","org.apache.catalina.Server");
digester 会在解析之后,返回一个对象。
5. 服务的中止
StandardServer.await是保持tomcat运行的秘密。方法启动一个 ServerSocket,侦听发出停止的字符串。这是一个死循环。当有停止运行的字符发出,跳出此循环。
Socket socket = null;
InputStream stream = null;
try {
    socket = serverSocket.accept();
    socket.setSoTimeout(10 * 1000);  // Ten seconds
    stream = socket.getInputStream();
} catch (AccessControlException ace) {
......
while (true) {
    ......
    StringBuffer command = new StringBuffer();
    ......
    while (expected > 0) {
        ......
        ch = stream.read();
        ......
         command.append((char) ch);
        ......
    }
    ......
    boolean match = command.toString().equals(shutdown);
    if (match)  break;
......
}
......
serverSocket.close();
跳出循环后,系统执行关闭连接等资源的操作,服务就中止了。
我们上面谈到,因为Catalina已经注册了关闭钩子,所以从命令行方式关闭进程也是可以释放资源的。但前提是JVM必须在。如果JVM都立刻杀掉了,释放的操作就不能进行了。
Catalina.stopServer方法用于发出一个让服务停止的指令
Socket socket = new Socket("127.0.0.1", server.getPort());
OutputStream stream = socket.getOutputStream();
String shutdown = server.getShutdown();
for (int i = 0; i < shutdown.length(); i++)
    stream.write(shutdown.charAt(i));
stream.flush();
stream.close();
socket.close();
这一篇内容就是这些。下一篇文章将介绍StandardService的工作过程,着重在Web容器的工作原理。

参考资料:
Tomcat用户手册
另外一个哥们的Tomcat源码分析
Tomcat源码
分享到:
评论

相关推荐

    蓝屏查看器

    &lt;span&gt;热搜词:&lt;/span&gt; &lt;a href="http://www.onlinedown.net/soft/50671.htm" target="_blank"&gt;360安全卫士&lt;/a&gt; &lt;a href="http://down2.uc.cn/pcbrowser/down.php?pid=4360" target="_blank"&gt;UC浏览器&lt;/a&gt; &lt;a ...

    条码枪GFS4470驱动

    &lt;span&gt;热搜词:&lt;/span&gt; &lt;a href="http://www.onlinedown.net/soft/50671.htm" target="_blank"&gt;360安全卫士&lt;/a&gt; &lt;a href="http://down2.uc.cn/pcbrowser/down.php?pid=4360" target="_blank"&gt;UC浏览器&lt;/a&gt; &lt;a ...

    <zt>Jquery Tutorial

    这部分内容是理解jQuery工作原理的关键,为后续的学习打下基础。 接下来的"万能的选择器"章节,讲解了jQuery如何通过选择器来选取DOM元素。jQuery支持CSS选择器,ID选择器,类选择器等多种方式,使得选取元素变得...

    众泰人才网完整无错生成HTML静态8.02版

    admin &lt;br&gt;管理员密码 admin&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt; 免费下载网址 http://www.ztsystem.cn/download/rc/ZT_RC_V802.rar&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt;&lt;br&gt; ╔------------------------------------------------╗&lt;br&gt; ...

    简单打地鼠游戏

    分钟&lt;br /&gt; 游戏倒计时:&lt;span id="djs"&gt;&lt;/span&gt;秒&lt;br /&gt; 游戏得分:&lt;span id="defen"&gt;&lt;/span&gt;&lt;br /&gt; &lt;button id="ksBtn"&gt; 开始游戏&lt;/button&gt; &lt;button id="ztBtn" &gt;暂停游戏&lt;/button&gt; &lt;button id="tzBtn"&gt; 停止...

    美淘网(淘宝)静态html页面

    4. **商品展示**:使用`&lt;div&gt;`分组元素展示商品,每个商品通常包含图片、标题、价格等信息,可以使用`&lt;img&gt;`标签插入图片,`&lt;h1&gt;`-`&lt;h6&gt;`定义标题等级,`&lt;p&gt;`添加描述文本。 5. **类别分类**:利用`&lt;section&gt;`和`...

    JS 中文简繁互相转换 有中文实例

    JS javascript 中文简繁互相转换 有中文实例&lt;br&gt;/*&lt;br&gt; * 简体繁体互转&lt;br&gt; * 例:&lt;br&gt; * var xxx=new SIMPLIF();&lt;br&gt; * var ft=xxx.unsimplified("你好啊,我爱你哦,天上人间");&lt;br&gt; * vat zt=xxx.simplify(ft)&lt;br...

    zt-process-killer:零周转流程杀手

    ...&lt; dependency&gt; &lt; groupId&gt;org.zeroturnaround&lt;/ groupId&gt; &lt; artifactId&gt;zt-process-killer&lt;/ artifactId&gt; &lt; version&gt;1.10&lt;/ version&gt;&lt;/ dependency&gt;...动机在Java中,不明确。 在Windows上,它会强制终止进程。 ...

    js时间控件(WdatePicker.js)

    这里基于js技术,是一款很好用的js时间控件。例如时间为期一个月代码如下: &lt;th&gt;开始日期 &lt;/th&gt; &lt;td&gt; &lt;input type="text" name="*" onFocus="WdatePicker({maxDate:'%y-%M-%d... &lt;span class="zt03"&gt;*&lt;/span&gt; &lt;/td&gt;

    zt-exec:ZeroTurnaround流程执行器

    &lt; artifactId&gt;zt-exec&lt;/ artifactId&gt; &lt; version&gt;1.12&lt;/ version&gt; &lt;/ dependency&gt; ... 动机 从Java运行外部进程时,可以采用多种方法。 有JRE选项,例如Runtime.exec()和ProcessBuilder 。 还有 。 尽管如此,我

    使用Java编写爬虫写入excel的多个sheet

    public static void writeDataToMultipleSheets(List&lt;List&lt;String&gt;&gt; data, String excelFilePath, List&lt;String&gt; sheetNames) throws IOException { Workbook workbook = new XSSFWorkbook(); // 创建一个新的Excel...

    html网页个人设计.zip_html5_plannedxw5_个人主页 html_个人网页_个人页html

    例如,`&lt;header&gt;`、`&lt;nav&gt;`、`&lt;section&gt;`、`&lt;article&gt;`和`&lt;footer&gt;`等语义化标签,增强了内容的结构和可读性;`&lt;canvas&gt;`元素为动态图形和交互式内容提供了画布;`&lt;video&gt;`和`&lt;audio&gt;`元素使得网页内嵌多媒体成为...

    (zt)Commons-logging + Log4j 入门指南

    &lt;artifactId&gt;commons-logging&lt;/artifactId&gt; &lt;version&gt;1.2&lt;/version&gt; &lt;/dependency&gt; &lt;dependency&gt; &lt;groupId&gt;log4j&lt;/groupId&gt; &lt;artifactId&gt;log4j&lt;/artifactId&gt; &lt;version&gt;1.2.17&lt;/version&gt; &lt;/dependency&gt; &lt;/...

    软考_系统分析师考试综合知识试题分类精解[上](1)

    系统分析师考试综合知识试题分类精解[上]&lt;br&gt;1-9章&lt;br&gt;(第1部分)&lt;br&gt;

    Java 爬虫jsoup简单的demo

    List&lt;LinkData&gt; dataList = new ArrayList&lt;&gt;(); Document doc = Jsoup.connect("http://example.com").get(); Elements links = doc.select("a"); for (Element link : links) { LinkData data = new ...

    ztdns:管理Zerotier网络成员的公共DNS记录

    零层DNS管理器 此工具旨在根据Zerotier网络的成员保持Route53 DNS记录的更新。 它获取成员列表,并获取具有.zt.的作用域Route53记录的列表.zt....zt_network= &lt; your&gt; sleep_timeout= &lt; a reasonable sl

    karaf使用配置文档

    2. `bundle:start &lt;id&gt;`:启动指定ID的bundle。 3. `feature:repo-add &lt;url&gt;`:添加一个新的feature仓库。 4. `feature:install &lt;feature-name&gt;`:安装指定的feature。 5. `config:list`:查看所有配置项。 6. `...

    html5+css网站

    1. **语义化元素**:HTML5引入了许多新的语义元素,如&lt;header&gt;、&lt;nav&gt;、&lt;article&gt;、&lt;section&gt;、&lt;aside&gt;和&lt;footer&gt;等,这些元素帮助开发者更好地结构化网页内容,提高可读性和可访问性。 2. **媒体元素**:HTML5的...

    zt云流量HTML响应式模板

    HTML5引入了许多新的标签,如&lt;header&gt;、&lt;nav&gt;、&lt;section&gt;等,它们有助于提高网页内容的结构化和可读性,同时增强了与JavaScript和CSS3的兼容性。这样的设计有利于搜索引擎优化(SEO),使网站更容易被搜索引擎识别和...

    marathon-compilers:编程竞赛的编译库。 该库可用于配置在线编译器或建立一个平台,以在编程比赛期间编译和运行程序

    马拉松编译器 持续集成 快速概述 创建该项目的目的是为了满足我们在举办的编程比赛期间对文件进行编译和执行的需要,... &lt; artifactId&gt;marathon-compilers&lt;/ artifactId&gt; &lt; version&gt;1.1&lt;/ version&gt; &lt;/ dependency&gt; 动

Global site tag (gtag.js) - Google Analytics