2019独角兽企业重金招聘Python工程师标准>>>

摘要: 在阅读tomcat源码前,我们一般都会有如下几个疑问: - web容器和servlet容器的区别是什么; - 在springMVC中的web.xml是什么时候加载到tomcat中的; - tomcat是怎么加载我们的web服务的; - tomcat是怎么实现的热部署; - 一个http请求

1.前言 1.1 问题思考 在阅读tomcat源码前,我们一般都会有如下几个疑问:

web容器和servlet容器的区别是什么; 在springMVC中的web.xml是什么时候加载到tomcat中的; tomcat是怎么加载我们的web服务的; tomcat是怎么实现的热部署; 一个http请求是怎么被tomcat监听到的,会有哪些处理; 为什么请求可以有需要通过nginx的,也可以不需要nginx的直接请求到tomcat上? …… 如果你想知道答案,那么接下来的文章会告诉你。

1.2 基本姿势 问题先放在一边,我们都知道Tomcat是一种web容器,用来接收http请求,并将请求得到的结果返回。那么如果要我们设计一个web容器,该怎么做?

很自然的会想到,要有一个网络连接通信维护者和一个请求处理者。通信维护者能够监听到请求并返回数据;请求处理者能够将请求进行分解处理,得到请求应该返回的数据。如果你的思路是这样的,那么恭喜你,你看源码会简单很多,因为Tomcat的设计核心就是这样的,由两大组件组成:Connector和Container。Connector负责的是底层的网络通信的实现,而Container负责的是上层servlet业务的实现。

阅读源码前的准备工作:tomcat源码下载,http://tomcat.apache.org/download-70.cgi 下载好源码后导入eclipse。Tomcat7.0工程是用ant构建的,在导入eclipse的时候可以选择file-new-project-java project from existing ant buildfile导入即可。

本文用的是tomcat7.0,原因是因为公司的tomcat是7.0,当然你也可以下tomcat9的源码进行分析。

也许你会猴急,希望快点知道Connector和Container是怎么设计和实现的,不过我要掉你胃口了,先得给你讲讲Tomcat启动过程,因为只有知道了Tomcat启动过程,才能对Connector和Container是怎么初始化的了然于胸,知道了Connector和Container的初始化才能准确的把握其结构。

2.tomcat启动 2.1 从main看起 启动命令行:java [****] org.apache.catalina.startup.Boostrap start,从命令行中可知,调用的Boostrap类的main方法,main方法参数是start。

public static void main(String args[]) {if (daemon == null) {Bootstrap bootstrap = new Bootstrap();try {bootstrap.init();}daemon = bootstrap;} else {Thread.currentThread().setContextClassLoader(daemon.catalinaLoader);}try {String command = "start";if (command.equals("startd")) {args[args.length - 1] = "start";daemon.load(args);daemon.start();} else if (command.equals("stopd")) {args[args.length - 1] = "stop";daemon.stop();} else if (command.equals("start")) {daemon.setAwait(true);daemon.load(args);daemon.start();} else if (command.equals("stop")) {daemon.stopServer(args);} else if (command.equals("configtest")) {daemon.load(args);if (null==daemon.getServer()) {System.exit(1);}System.exit(0);} }}

从源码中可以看到,main方法可传入的参数有 startd、stopd、start、stop,startd和start的区别是设置了一个await的bool变量为true,这点在后面会继续提到。 通过运行tomcat参数的可以看到main方法主要做了3件事:

实例化Boostrap对象,并调用其init方法; 调用load方法 调用start方法; 2.2、实例化Boostrap对象: 调用Boostrap的init方法主要完成三件事: -(1)初始化路径:设置HOME和Base路径,其中HOME是tomcat安装目录,Base是tomcat工作目录;如果我们想要运行Tomcat 的 多个实例,但是不想安装多个Tomcat软件副本。那么我们可以配置多个工作 目录,每个运行实例独占一个工作目录,但是共享同一个安装目录。 -(2)初始化类加载器:初始化Tomcat类加载器:commonLoader、catalinaLoader、sharedLoader commonLoader无父加载器,catalinaLoader和sharedLoader的父加载器都是commonLoader,其中若tomcat的配置文件没有配置:server.loader则catalinaLoader=commonLoader,同理,没配置shared.loader……,这三种都是URLClassLoader,使用Java 中的安全模型。 Tomcat 的类加载器体系如下结构所示: screenshot

-(3)初始化Boostrap的Catalina对象:通过反射生成Catalina对象,并通过反射调用setParentClassLoader方法设置其父 ClassLoader为sharedLoader。为什么要用反射,不直接在声明的时候生成对象?使用反射来生成实例的原因是因为在tomcat的发展历史中可以不止Catalina一种启动方式,现在看代码已经没必要了。 -(4)其他:主线程的classLoader设置为catalinaLoader,安全管理的classLoad设置为catalineLoader。

public void init() throws Exception { //设置系统变量CATALINA_HOME和CATALINA_BASE setCatalinaHome(); setCatalinaBase(); //初始化classLoader initClassLoaders();

    Thread.currentThread().setContextClassLoader(catalinaLoader);//主线程的classLoaderSecurityClassLoad.securityClassLoad(catalinaLoader);// Load our startup class and call its process() methodif (log.isDebugEnabled())log.debug("Loading startup class");Class<?> startupClass =catalinaLoader.loadClass("org.apache.catalina.startup.Catalina");Object startupInstance = startupClass.newInstance();//TODO:使用反射来生成实例的原因是因为在tomcat的发展历史中可以不止Catalina一种启动方式,现在看代码已经没必要了// Set the shared extensions class loaderif (log.isDebugEnabled())log.debug("Setting startup class properties");String methodName = "setParentClassLoader";Class<?> paramTypes[] = new Class[1];paramTypes[0] = Class.forName("java.lang.ClassLoader");Object paramValues[] = new Object[1];paramValues[0] = sharedLoader;Method method =startupInstance.getClass().getMethod(methodName, paramTypes);method.invoke(startupInstance, paramValues);//TODO:设置catalina的父classLoader为sharedLoader        catalinaDaemon = startupInstance;}

2.3、Boostrap的load方法 调用Catalina的load方法。通过参数判断设置一些条件并判断是否立即加载,加载调用load方法主要做了4件事: -(1)判断系统变量catalina设置HOME和Base路径是否设置,没有的话则设置一下(感觉像是烂代码啊,在Boostrap初始化的时候不是已经设置过一次吗?为什么不写在一起?) -(2)初始化命名服务的基本配置 -(3)Digester类,默认将conf/server.xml解析成相应的对象,这个解析很重要,因为是他完成了Connector和Container的初始化工作。先mark下,接下来会详细的介绍Digester是怎么完成Connector和Container的初始化,现在我们只要知道是这里完成的就可以了。 -(4)Server调用init初始化声明周期,其父类LifecycleMBeanBase实现

//去掉了一些无用的代码 public void load() {

    long t1 = System.nanoTime();// CATALINA_BASE和CATALINA_HOME设置initDirs();// 初始化命名服务的基本配置initNaming();//Digester类,主要用于处理xml配置文件,将xml文件转换成对应的java对象(默认为conf/server.xml)  Digester digester = createStartDigester();InputSource inputSource = null;InputStream inputStream = null;File file = null;try {try {// 配置文件,由命令行参数-config指定,否则取默认值conf/server.xml  file = configFile();inputStream = new FileInputStream(file);inputSource = new InputSource(file.toURI().toURL().toString());} try {inputSource.setByteStream(inputStream);//将cataline对象压栈,如果栈为空,则设置root为cataline对象digester.push(this);//遇到相应的应用对对象赋值,相应的调用则调用digester.parse(inputSource);}} getServer().setCatalina(this);// Stream redirectioninitStreams();// Start the new servertry {getServer().init();} ……
}

2.4、 Boostrap的start方法 调用Catalina的start方法: -(1)Server加载:Server才是正真的tomcat服务执行者,包括Connector和Container。调用load方法; -(2)调用Server的start方法,最终调用的是StandardServer的startInternal方法,调用自己所有的Service的start方法,启动connector和container、excutor的start方法,后文会继续扩展。 -(3)注册钩子

public void start() { try { //服务启动 getServer().start(); } …… // Register shutdown hook if (useShutdownHook) { if (shutdownHook == null) { shutdownHook = new CatalinaShutdownHook(); } Runtime.getRuntime().addShutdownHook(shutdownHook); LogManager logManager = LogManager.getLogManager(); if (logManager instanceof ClassLoaderLogManager) { ((ClassLoaderLogManager) logManager).setUseShutdownHook( false); } }

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

可以看到,Boostrap调用Catalina的方法时,全部都用的是反射,包括生成Catalina对象。而tomcat的启动说白了其实就是初始化并启动Connector和Container。

转载于:https://my.oschina.net/u/3229047/blog/1355195

tomcat源码分析--初始化与启动相关推荐

  1. Tomcat源码分析(九)--Session管理

    本系列转载自 http://blog.csdn.net/haitao111313/article/category/1179996 在明白Tomcat的Session机制之前,先要了解Session, ...

  2. Tomcat源码分析(十)--部署器 转载

    本系列转载自 http://blog.csdn.net/haitao111313/article/category/1179996 我们知道,在Tomcat的世界里,一个Host容器代表一个虚机器资源 ...

  3. Tomcat源码分析--转

    一.架构 下面谈谈我对Tomcat架构的理解 总体架构: 1.面向组件架构 2.基于JMX 3.事件侦听 1)面向组件架构 tomcat代码看似很庞大,但从结构上看却很清晰和简单,它主要由一堆组件组成 ...

  4. Tomcat源码分析

    Tomcat源码分析 ` 最近深入了解了下tomcat的源码,在此记录下 文章目录 Tomcat源码分析 前言 一.Tomcat整体架构是什么? 1)分析配置文件server.xml, 2)网上盗个图 ...

  5. srs源码分析3-srs的启动

    本文分析的srs版本是0.6.0 srs源码分析1-搭建环境 srs源码分析2-浅析state_threads srs源码分析3-srs的启动 srs源码分析4-客户端的连接 srs源码分析5-han ...

  6. live555 源码分析:播放启动

    本文分析 live555 中,流媒体播放启动,数据开始通过 RTP/RTCP 传输的过程. 如我们在 live555 源码分析:子会话 SETUP 中看到的,一个流媒体子会话的播放启动,由 Strea ...

  7. Tomcat源码分析——server.xml文件的加载

    前言 作为Java程序员,对于tomcat的server.xml想必都不陌生.本文基于Tomcat7.0的Java源码,对server.xml文件是如何加载的进行分析. 源码分析 Bootstrap的 ...

  8. Springboot源码分析之内嵌tomcat源码分析

    Springboot源码是内嵌tomcat的,这个和完整的tomcat还是不同. 内嵌tomcat的源码在tomcat-embed-core等3个jar包里 展开tomcat-embed-core的c ...

  9. android6.0源码分析之Activity启动过程

    Activity最为Android开发者最熟悉的组件,由ActivityManagerService服务进行调度管理,而ActivityManagerService的启动过程在activitymana ...

最新文章

  1. [3]MVC3中使用Telerik(转)
  2. java制作带有logo的二维码,解决zxing中文乱码
  3. [leetcode]151.翻转字符串里的单词
  4. 【英语学习】【医学】Unit 09 The Respiratory System
  5. POJ2823 Sliding Window 单调队列
  6. html+加粗+w3c,HTML5教程:html标签属性通过w3c验证
  7. 基于Java的网络教学系统
  8. 寻找隐藏在CloudFlare和Tor后的真实IP
  9. 【Javafx】关于属性绑定需要在动画设置之后
  10. 2018web前端不好找工作之web前端面试简历补救篇
  11. Xshell快速命令集解放生产力
  12. 公众号开发素材管理,php删除微信素材
  13. Auto.js加密解密|快照Snapshot解密|Dex转js|Auto.js庖丁下载|解除APP限制
  14. Java——万字总结网络编程
  15. 紫色小人_HaPPYBoY(LoVE)
  16. SSM与Spring
  17. cartographer建图,重定位及发布消息结构为nav_msgs::Odometry的odom话题
  18. 0.linux中英文环境的切换
  19. 听诊器的基本构造及其特征
  20. PaddlePaddle21天深度学习训练营学习心得

热门文章

  1. HTML5等先关。。。
  2. 关于使用jacob出现的异常
  3. python解决高并发的方法
  4. Linux文件系统变成只读的解决方法
  5. python 常见的异常类型
  6. 记一次网站无法访问解决过程,服务器80端口问题解决过程
  7. win11资源管理器卡顿怎么办 Windows11解决资源管理器卡顿的步骤方法
  8. activiti并行网关
  9. VMware Mac 全屏问题
  10. linux 自带多路径工具,RHEL6使用系统自带多路径软件配置多路径