《深入剖析Tomcat》读书笔记(二)
三、容器Container
Container 是容器的父接口,所有子容器都必须实现这个接口。
Container 容器的设计用的是典型的责任链的设计模式,它有四个子容器组件构成,分别是:Engine、Host、Context、Wrapper,这四个组件不是平行的,而是父子关系,Engine 包含 Host,Host 包含 Context,Context 包含 Wrapper。通常一个 Servlet class 对应一个 Wrapper,如果有多个 Servlet 就可以定义多个 Wrapper。
| 容器定义 | 标准实现类 | 基础阀类 | 其他 |
|
Engine: 表示整个Catalina servlet引擎 |
StandardEngine | StandardEngineValue |
使用映射器Mapper返回子容器中特定的Host |
|
Host: 表示包含一个或者多个Context容器的虚拟主机 |
StandardHost | StandardHostValue | 使用映射器Mapper返回子容器中特定的context |
|
Context: 表示一个Web应用程序 |
StandardContext | StandardContextValue | 使用映射器Mapper返回子容器中特定的Wrapper |
|
Wrapper: 表示一个独立的servlet |
StandardWrapper | StandardWrapperValue |
调用其invoke方法之前先要执行过滤器链 会调用具体servlet的service(),该service默认是无状态可并发的 |

四、容器Container如何处理请求
我们现在解决传递到container的请求是如何找到对应的servlet的,首先关注connector启动的时候Container的初始化流程:
public static void main(String[] args) {
HttpConnector connector = new HttpConnector();
//两个wapper
Wrapper wrapper1 = new SimpleWrapper();
wrapper1.setName("Primitive");
wrapper1.setServletClass("PrimitiveServlet");
Wrapper wrapper2 = new SimpleWrapper();
wrapper2.setName("Modern");
wrapper2.setServletClass("ModernServlet");
//建立context和wapper的关系
Context context = new SimpleContext();
context.addChild(wrapper1);
context.addChild(wrapper2);
//建立两个阀,并放到context的管道中
Valve valve1 = new HeaderLoggerValve();
Valve valve2 = new ClientIPLoggerValve();
((Pipeline) context).addValve(valve1);
((Pipeline) context).addValve(valve2);
// 建立Mapper,用于映射wrapper
// context.addServletMapping(pattern, name);
Mapper mapper = new SimpleContextMapper();
mapper.setProtocol("http");
context.addMapper(mapper);
context.addServletMapping("/Primitive", "Primitive");
context.addServletMapping("/Modern", "Modern");
//建立class loader
Loader loader = new SimpleLoader();
context.setLoader(loader); //建立连接器和容器的关系
connector.setContainer(context); try {
connector.initialize(); //启动连接器
connector.start();
}
catch (Exception e) { e.printStackTrace(); }
}
}
1、管道Pipeline:我们这里看到了Pipeline,可以理解为Context容器内部维护了一个管道,可以向这个管道中添加各式各样的阀。
当invoke调用到达context容器后,下面实际调用的是Pipeline的invoke方法,该方法依次遍历管道上的各种阀,一个阀处理完成后,交个下一个阀进行处理,该设计模式为责任链模式。基础阀总是随后一个执行,基础阀一般用来找到对应的子容器warpper并调用子容器的service()方法。
2、映射器Mapper:那么是如何找到对应的servlet的呢,这里用到了映射器Mapper,context根据映射规则就可以找到对应的servlet,执行对应的service(),至此,一个处理请求完毕。
3、总体过一遍:连接器收到请求后,创建request/response,调用StandardContext的invoke方法;该invoke方法调用其管道对象的invoke方法,管道对象的invoke方法会依次调用其阀的invoke,最后调用基础阀的invoke;基础阀是StandardContextValue类的实例,基础阀的invoke方法被调用,基础阀invoke会获取Wrapper实例,并依次执行Warpper的管道对象的阀,最终通过Mapper找到对应的servlet,执行servlet的service()方法。
五、容器Container生命周期管理Lifecycle
catalina允许一个组件包含多个其他组件,当父组件启动和关闭的时候,子组件会跟着一起启动和关闭,该部分关键是Lifecycle接口
public interface Lifecycle {
public void addLifecycleListener(LifecycleListener listener);
public void removeLifecycleListener(LifecycleListener listener);
public void start() throws LifecycleException;
public void stop() throws LifecycleException;
}
1、启动/关闭的时候,会依次遍历注册到该容器的子容器,调用子容器的start/stop方法,像链式爆炸一样,把整个关联的容器都启动/关闭
2、事件Listener会预先调用容器的addLifecycleListener方法,将事件监听器加到该容器维护的监听器数组中,于是容器在特定的时机调用 lifecycle.fireLifecycleEvent(START_EVENT, null);就会遍历这些监听器,调用这些监听器的lifecycleEvent()方法,这是一种典型的发布者订阅者模式的应用。
六、载入器Loader
tomcat使用自定义载入器的原因:
1、servlet应该只允许载入web-inf/classes、web-inf/lib下的类,访问其余目录(环境变量中classpath指定的路径)会是危险的,因此要设置一些载入规则。
2、将载入的类缓存起来
3、支持自动重载的功能。当web-inf/classes、web-inf/lib下的类发生变化的时候,web应用会重新载入这些类。tomcat实现该功能的方法是使用一个额外的线程不断地检查这些类的时间戳,发现变动后会自动载入。
支持自动重载功能,需要设置server.xml
<Context docBase="D:\ProgramServer\Tomcat7\stswebapps\traveller" path="/traveller" reloadable="true" source="org.eclipse.jst.jee.server:traveller"/>
org.apache.catalina.loader.webappLoader类实现Loader接口,作为应用载入器,负责载入web应用程序中使用到的类。
WebappLoader类会创建org.apache.catalina.loader.WebappClassLoader类的实例作为类载入器,WebappClassLoader默认会使用java.net.URL.ClassLoader具体承担载入类的工作。
WebappClassLoader会缓存它所加载的类,它会维护一个Vector对象,保存已经载入的类,防止这些类在不适用的时候被当做垃圾回收。
七、Session管理器
1、session对象怎么存储的
Context容器的Session管理器会管理Context容器中所有活动的Session对象,这些活动的Session对象都存储在一个名为sessions的HashMap变量中
2、session对象超时失效是怎么做到的
Session管理器通过建立一个专门的线程,来实现销毁已经失效的Session对象的。其思想就是周期地遍历Session管理器管理的所有Session对象,将Session对象的lastAccessedTime和当前时间进行比较,如果两者的差值超过了某个常数,就会调用Session对象的expire()方法使其失效。
3、session对象的持久化
session管理器可以将session对象持久化,存储到文件存储器或者通过JDBC存储到数据库中。以便再次启动的时候重新读入内存,维持session的状态不变化。
在持久化session管理器中,session可以备份,也可以换出。当备份一个session对象的时候,该Session对象会被复制到存储器中,而原对象依然保留在内存中。当对象被换出,意思是当前活动的session对象超过了上限值,或者这个session对象闲置了过长的时间,之所以换出是为了节省时间。
因为session对象可以被换出,所以它既可能存储在内存中,也可能存储在存储器中,因此,当客户端需要访问session数据的时候,findSession()方法会在内存中查找是否存在该session实例,如果没有找到,就到存储器中去查找。
public Session findSession(String id) throws IOException {
Session session = super.findSession(id);
if (session != null)
return (session);
// See if the Session is in the Store
session = swapIn(id);
return (session);
}
《深入剖析Tomcat》读书笔记(二)的更多相关文章
- 《CSS世界》笔记二:盒模型四大家族
上一篇:<CSS世界>笔记一:流/元素/尺寸下一篇:<CSS世界>笔记三:内联元素与对齐 写在前面 在读<CSS世界>第四章之前,粗浅的认为盒模型无非是margin ...
- CSS揭秘读书笔记 (一)
CSS揭秘读书笔记 (一) 一.半透明边框 要想实现半透明边框可以使用border: border: 10px solid hsla(0,0%,100%,.5); background: ...
- 《你必须知道的.NET》读书笔记二:小OO有大原则
此篇已收录至<你必须知道的.Net>读书笔记目录贴,点击访问该目录可以获取更多内容. 一.单一职责原则 (1)核心思想:一个类最好只做一件事,只有一个引起它变化的原因 (2)常用模式:Fa ...
- spring揭秘 读书笔记 二 BeanFactory的对象注册与依赖绑定
本文是王福强所著<<spring揭秘>>一书的读书笔记 我们前面就说过,Spring的IoC容器时一个IoC Service Provider,而且IoC Service Pr ...
- ES6读书笔记(二)
前言 前段时间整理了ES6的读书笔记:<ES6读书笔记(一)>,现在为第二篇,本篇内容包括: 一.数组扩展 二.对象扩展 三.函数扩展 四.Set和Map数据结构 五.Reflect 本文 ...
- 《精通CSS》读书笔记(一)
最近新添16本书,目前开始看陈剑瓯翻译的<精通CSS——高级Web标准解决方案>(Andy Budd, CSS Mastery -- Advanced Web Standards Solu ...
- spring揭秘 读书笔记 二 BeanFactory的对象注冊与依赖绑定
本文是王福强所著<<spring揭秘>>一书的读书笔记 我们前面就说过,Spring的IoC容器时一个IoC Service Provider,并且IoC Service Pr ...
- 【记】《.net之美》之读书笔记(二) C#中的泛型
前言 上一篇读书笔记,很多小伙伴说这本书很不错,所以趁着国庆假期,继续我的读书之旅,来跟随书中作者一起温习并掌握第二章的内容吧. 一.理解泛型 1.为什么要使用泛型?-----通过使用泛型,可以极大地 ...
- Mastering Web Application Development with AngularJS 读书笔记(二)
第一章笔记 (二) 一.scopes的层级和事件系统(the eventing system) 在层级中管理的scopes可以被用做事件总线.AngularJS 允许我们去传播已经命名的事件用一种有效 ...
- how tomcat works 读书笔记(二)----------一个简单的servlet容器
app1 (建议读者在看本章之前,先看how tomcat works 读书笔记(一)----------一个简单的web服务器 http://blog.csdn.net/dlf123321/arti ...
随机推荐
- 证据对抗、证据链标准 z
证据链乃诉讼中应对证据冲突的法宝 大部分当事人知道“证据链”这一诉讼专用术语,但几乎所有当事人将“证据链”理解成了人证.书证.物证,鉴定结论等证据的组合,将“证据链”视为不同种类证据依次出击的“组 ...
- Android NFC开发概述
NFC手机相比普通手机来说,有以下3个附加功能: 1.可以当成POS机来用,也就是“读取”模式 2.可以当成一张卡来刷,也就是NFC技术最核心的移动支付功能 3.可以像蓝牙.Wi-Fi一样做点 ...
- 【转】group by多个字段理解
来源:http://uule.iteye.com/blog/1569262 首先group by 的简单说明: group by 一般和聚合函数一起使用才有意义,比如 count sum avg等,使 ...
- trim
这是一个很常见的函数,他的所用是去掉字符序列左边和右边的空格,如字符串str = " ai lafu yo ";str = trim(str); cout << str ...
- javascript最新深度克隆对象方法
javascript最新深度克隆对象方法 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" &qu ...
- [ActionScript 3.0] AS3 GUID(全局唯一标识符)
package com.controls { import flash.display.Sprite; import flash.system.Capabilities; public class G ...
- topngroupcollector
分类的字段int f1 /* key: f1 * value: doc * size: top n */ map<int, doc>() if(map.size==n) buildOrde ...
- DevExpress某些控件继承后的可编辑性
今天在使用DevExpress的BarManager菜单控件时,发现在进行继承时无法在继承的子类窗体中对其进行编辑与修改,另外像GridView也有类似的情形,后来查阅资料后,现在可通过DevExpr ...
- cocos2dx 锁定30帧设置
修改代码: AppDelegate.cpp // set FPS. the default value is 1.0/60 if you don't call this pDirector->s ...
- CDN和DNS
相信有很多的朋友会被这几个名词绕的有些头大,很多朋友觉得智能DNS跟双线加速.CDN加速是类似的技术.其实不然,虽然他们的目的都是一个:让用户更快的访问网站.但是他们的应用原理却大相径庭.大家一定很清 ...