从开始学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源码
分享到:
相关推荐
<span>热搜词:</span> <a href="http://www.onlinedown.net/soft/50671.htm" target="_blank">360安全卫士</a> <a href="http://down2.uc.cn/pcbrowser/down.php?pid=4360" target="_blank">UC浏览器</a> <a ...
<span>热搜词:</span> <a href="http://www.onlinedown.net/soft/50671.htm" target="_blank">360安全卫士</a> <a href="http://down2.uc.cn/pcbrowser/down.php?pid=4360" target="_blank">UC浏览器</a> <a ...
NULL 博文链接:https://tianjun309.iteye.com/blog/1072285
admin <br>管理员密码 admin<br><br><br><br><br><br><br><br> 免费下载网址 http://www.ztsystem.cn/download/rc/ZT_RC_V802.rar<br><br><br><br><br> ╔------------------------------------------------╗<br> ...
分钟<br /> 游戏倒计时:<span id="djs"></span>秒<br /> 游戏得分:<span id="defen"></span><br /> <button id="ksBtn"> 开始游戏</button> <button id="ztBtn" >暂停游戏</button> <button id="tzBtn"> 停止...
JS javascript 中文简繁互相转换 有中文实例<br>/*<br> * 简体繁体互转<br> * 例:<br> * var xxx=new SIMPLIF();<br> * var ft=xxx.unsimplified("你好啊,我爱你哦,天上人间");<br> * vat zt=xxx.simplify(ft)<br...
...< dependency> < groupId>org.zeroturnaround</ groupId> < artifactId>zt-process-killer</ artifactId> < version>1.10</ version></ dependency>...动机在Java中,不明确。 在Windows上,它会强制终止进程。 ...
这里基于js技术,是一款很好用的js时间控件。例如时间为期一个月代码如下: <th>开始日期 </th> <td> <input type="text" name="*" onFocus="WdatePicker({maxDate:'%y-%M-%d... <span class="zt03">*</span> </td>
< artifactId>zt-exec</ artifactId> < version>1.12</ version> </ dependency> ... 动机 从Java运行外部进程时,可以采用多种方法。 有JRE选项,例如Runtime.exec()和ProcessBuilder 。 还有 。 尽管如此,我
系统分析师考试综合知识试题分类精解[上]<br>1-9章<br>(第1部分)<br>
零层DNS管理器 此工具旨在根据Zerotier网络的成员保持Route53 DNS记录的更新。 它获取成员列表,并获取具有.zt.的作用域Route53记录的列表.zt....zt_network= < your> sleep_timeout= < a reasonable sl
马拉松编译器 持续集成 快速概述 创建该项目的目的是为了满足我们在举办的编程比赛期间对文件进行编译和执行的需要,... < artifactId>marathon-compilers</ artifactId> < version>1.1</ version> </ dependency> 动
The thermoelectric performance of DNA duplex chains (G:C)–(T:A)n–(G:C)3 is studied by a tight-binding<br>ladder ... Big figure<br>of merit (ZT) might be available for large n. This result is applica
ArrayList<student> arr=new ArrayList <student>();boolean zt=false; while(true){ if(zt==false){ System.out.println("1.输入数据");} else{System.out.println("1输入数据 0显示");} int input=sc.nextInt();...
例如: 代码如下: <li><...</li> 其中name属性值肯定是一样的,接下来用js获取选中的值: 代码如下: var zt = document.getElementsByName(“zt”); for(var i=0;i<zt.length;i++){ if(zt[i].c
<div class="zt"> <form method="post" action="" name="form"> <div class="biaoti"><span>计 算 器</span><hr/></div> <div> <input name="ipt" type="text" id="ipts" value="0" size="40" ...
ArrayList<Yuangong> arr=new ArrayList<Yuangong>(); boolean zt=false; Yuangong yg=new Yuangong(); while(true){ if(zt==false){ System.out.println("1.输入数据");} else{System.out.println("1输入数据 ...
Zebra ZT410(产品特点:分辨率: 203 dpi/每毫米 8 点...) Zebra ZT420(产品特点:打印宽度: 6.6 英寸/168 毫米...) Zebra ZT210(产品特点:预装中文字体,12 点/毫米/300dpi 多选 ...) Zebra ZT230(产品...
赠送jar包:zt-exec-1.9.jar; 赠送原API文档:zt-exec-1.9-javadoc.jar; 赠送源代码:zt-exec-1.9-sources.jar; 赠送Maven依赖信息文件:zt-exec-1.9.pom; 包含翻译后的API文档:zt-exec-1.9-javadoc-API文档-...
这款RS485通信芯片zt13085e用的不多,网上很少这款的原理图库和PCB库,自己画了一个。