第一个
有一个养鸡场几千只鸡分别排列的被关在狭小的鸡笼里鸡笼的前面有传送带为小鸡送来食物后面有传送带送走产下的鸡蛋可是却注意到有几十只小鸡在外面逍遥还有饲养员在给他们喂食
于是就问是否需要帮忙吧这些鸡赶到笼子里去呢?
饲养员回答说哦这些鸡是在外面养着的要是笼子里的那些鸡看不到有的鸡自由自在的生活它们就会失去信心不下蛋没有这些放养的鸡其它的就会放弃一切然后死去
突然惊异的发现我们的生活方式和这些鸡是何等的相似有多少人生活在笼子里看着外面的世界里别人在探险在实现自己的梦想自由自在的生活然后继续省会在自己的笼子里
作者描述自己起先放弃一年的时间去到外面的世界看看最后放弃了自己的职业成为了逍遥自在的那一个
第二个故事我放到本文的最后告诉各位哦哈哈
主题
在前面的主题中我们已经了解过servlet的工作流程Servlet我们到底能走多远系列()
问题:tomcat什么时候加载servlet?
有两种情况
一种是启动时加载
一种是请求时加载
第一种是在webxml中的<servlet>节点下增加类似<loadonstartup></loadonstartup>的节点
例子如下
<servlet>
<servletname>DicDataIniter</servletname>
<servletclass>cominitDicDataIniter</servletclass>
<loadonstartup></loadonstartup> </servlet> <servletmapping>
<servletname>DicDataIniter</servletname>
<urlpattern>/DicDataIniter</urlpattern> </servletmapping>
关于loadonstartup需要了解的是
)loadonstartup元素标记容器是否在启动的时候就加载这个servlet(实例化并调用其init()方法))它的值必须是一个整数表示servlet应该被载入的顺序)当值为或者大于时表示容器在应用启动时就加载并初始化这个servlet;)当值小于或者没有指定时则表示容器在该servlet被选择时才会去加载)正数的值越小该servlet的优先级越高应用启动时就越先加载)当值相同时容器就会自己选择顺序来加载所以<loadonstartup>x</loadonstartup>中x的取值代表的是优先级
第二种的流程类似如下图
问题:tomcat是怎么加载一个servlet类的呢?
根据上篇对一个简单的webServer模拟(How Tomcat Works 学习我们到底能走多远系列())在到达加载servlet类实例化一个servlet实例这一步之前的工作我们已经完成了我们已经取得了来自http消息中的uri也就已经取得了需要调用的servlet的名字
根据《How tomcat works》先搞一个自己的servlet:没什么逻辑只是生成class文件放到指定的文件夹下等待加载
package codetomcatservletContainer;import javaioIOException;import javaioPrintWriter;import javaxservlet*;public class PrimitiveServlet implements Servlet{
public void init(ServletConfig config) throws ServletException {
Systemoutprint(init)
}
public ServletConfig getServletConfig() {
return null;
}
// 一旦实现了service方法那么就不会就不会调用父类的service方法从而调用doGet或doPost
public void service(ServletRequest req ServletResponse res)
throws ServletException IOException {
Systemoutprint(service)
PrintWriter writer = resgetWriter()
writerprintln(Hello Roses are red)
}
public String getServletInfo() {
return null;
}
public void destroy() {
Systemoutprint(destroy)
}}
加载方法如下
package codetomcatservletContainer;import javaioFile;import javaioIOException;import MalformedURLException;import URL;import URLClassLoader;import URLStreamHandler;import javaxservletServlet;import javaxservletServletRequest;import javaxservletServletResponse;public class ServletProcessor {
public void process(Request request Response response){
// 取得request处理好的uri
String uri = requestgetUri()
// servlet 类名
String servletName = urisubstring(uriindexOf(/) + )
// 类加载器利用提供的url目录来加载class文件大多数情况下我们的程序在jvm启动的已经有类加载器加载我们需要的类文件
// 但是最为需要及时部署的服务器servlet容器需要自己去加载放到服务器的工程
URLClassLoader loader = null;
// URLClassLoader构造函数需要URL数组这里是需要加载一个类
URL[] urls = new URL[];
// 构造URL用虽然是null但如果直接在URL中直接用null来代替编译器不答应啊
URLStreamHandler streamHandler = null;
File classPath = new File(/webroot)
try {
// servlet文件夹位置
String repository = (new URL(file null classPathgetCanonicalPath() + Fileseparator))toString()
urls[] = new URL(null repository streamHandler)
// 得到传说中的类加载器
loader = new URLClassLoader(urls)
} catch (MalformedURLException e) {
eprintStackTrace()
} catch (IOException e) {
eprintStackTrace()
}
// class文件加载后就是java中的Class类实例啦
Class myClass = null;
try {
// servlet Class类实例
myClass = loaderloadClass(servletName)
} catch (ClassNotFoundException e) {
eprintStackTrace()
}
Servlet servlet = null;
try {
// 用newInstance方法得到一个servlet实例
servlet = (Servlet)myClassnewInstance()
// 调用PrimitiveServlet的service方法
servletservice((ServletRequest)request (ServletResponse)response)
} catch (Exception e) {
Systemoutprintln(etoString())
}
}}
如此servlet流程中的初始加载这一步就可以勉强完成了tomcat中的实现要复杂很多吧原理基本是差不多了先打下基础
关于类加载
其实tomcat启动的时候就有大量的类加载工作需要做应为tomcat本身就是个java工程嘛
SystemgetProperty:
有个类似写日志的方法
/**
* 写日志
* @param logString
* @throws IOException
*/
public void writeLog(String logString) throws IOException {
File logFile = new File(d:\\我的文档\\testlog)
// 写
FileWriter writer = new FileWriter(logFiletrue)
// 取得回车符号
String nextLine = SystemgetProperty(lineseparator)
writerwrite(logString + nextLine)
writerflush()
writerclose()
}
其中SystemgetProperty(lineseparator)方法值得熟悉
我们可以通过这样的方式取得一些System级别的常量
javaversion Java 运行时环境版本javavendor Java 运行时环境供应商javavendorurl Java 供应商的 URLjavahome Java 安装目录javavmspecificationversion Java 虚拟机规范版本javavmspecificationvendor Java 虚拟机规范供应商javavmspecificationname Java 虚拟机规范名称javavmversion Java 虚拟机实现版本javavmvendor Java 虚拟机实现供应商javavmname Java 虚拟机实现名称javaspecificationversion Java 运行时环境规范版本javaspecificationvendor Java 运行时环境规范供应商javaspecificationname Java 运行时环境规范名称javaclassversion Java 类格式版本号javaclasspath Java 类路径javalibrarypath 加载库时搜索的路径列表javaiotmpdir 默认的临时文件路径piler 要使用的 JIT 编译器的名称javaextdirs 一个或多个扩展目录的路径osname 操作系统的名称osarch 操作系统的架构osversion 操作系统的版本fileseparator 文件分隔符(在 UNIX 系统中是/)pathseparator 路径分隔符(在 UNIX 系统中是:)lineseparator 行分隔符(在 UNIX 系统中是/n)username 用户的账户名称userhome 用户的主目录userdir 用户的当前工作目录
好吧来讲一下第二个故事吧
一个印第安人和他的朋友走到美国繁华的街道上汽车的鸣笛声警笛声人们的喊声整个街道噪声简直震耳欲聋
突然一个印第安人说我听到有个蟋蟀在叫
他的朋友说什么?你疯了吗?在这么吵的地方你怎么可能听到蟋蟀的叫声呢?
可是那个印第安人却非常肯定他走到街对面的一块水泥石板前上面长满了灌木他在灌木堆里找就在那里找到了一只蟋蟀
他的朋友问太不可思议了你是怎么做到的?
印第安人说没有什么不可思议的着取决于什么对你是真正重要的来我来告诉你怎么回事
他拿出一块硬币扔到人行道上他们发现几乎所有在人行道上的人都低头去查看地上的硬币是不是他们遗落的
你明白了吗?这一切取决于什么对你来说是最重要的
让我们继续前行
努力不一定成功但不努力肯定不会成功
共勉