《深入剖析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 ...
随机推荐
- JAVA集合小结
下面是我自己画的,关系画得没上面好,但我自己看着清楚些 还有一张下载来的: 有序否 允许元素重复否 Collection 否 是 List 是 是 Set AbstractSet 否 否 Hash ...
- SqlServer2005基于已有表创建分区
随着当今数据库的容量越来越快的朝着在大型数据库或超大型数据库的发展,对于数据库中的大 型表以及具有各种访问模式的表的可伸缩性和可管理性运行环境变得尤为重要, SQL server 从 SQL serv ...
- bootstrap小例子等
一个简单的表单样式: <div class="row"> <form action="#" class="form-horizont ...
- *(volatile unsigned long *) 语法
#define GPJ2CON (*(volatile unsigned long *) 0xE0200280) 开始对于上面的语法不太理解,但通过拆分,我们可以看的更明白: 1.(unsigned ...
- [技巧]如何清除VS2008的最近打开项目
)删除最近打开的文件 运行regedit,打开HKEY_CURRENT_USER\Software\Microsoft\VisualStudio\9.0\FileMRUList 之后,在右边删除相应键 ...
- Oracle 的过程与函数
一.过程 1 .过程创建和调用 过程 (procedure) 是一个 PL/SQL 语句块,它存储在数据字典中并可被应用程序调用.可以使用过程存储数据库中频繁使用的应用逻辑.当执行一个过程时,其语句被 ...
- 类成员函数指针 ->*语法剖析
在cocos2d-x中,经常会出现这样的调用,如 ->*,这个是什么意思呢,如下面得这个例子: , 其实这是对类的成员函数指针的调用,在cocos2dx中,这种形式多用于回调函数的调用.如我们经 ...
- windows Phone 浏览器窗口的尺寸
移动设备的屏幕一般都比PC小很多,移动设备的浏览器会将一个较大的 “虚拟” 窗口映射到移动设备的屏幕上,然后按一定的比例(3:1或2:1)进行缩放.也就是说当我们加载一个普通网页的时候,移动浏览器 ...
- MFC创建非模态对话框并修改CStatic文字
//由IDD_STATUS_DIALOG创建一个对话框CDialog* pCheckNetStatusDlg = new CDialog(); pCheckNetStatusDlg->Creat ...
- Yii 中比较常用的rules验证规则记录
查看代码 打印 01 return array( 02 03 //必须填写 04 array('email, username, password,agree,verifyPa ...