码迷,mamicode.com
首页 > 其他好文 > 详细

Tomcat剖析(二)

时间:2015-09-07 00:30:22      阅读:280      评论:0      收藏:0      [点我收藏+]

标签:

Tomcat剖析(二)

目录

  1. Tomcat剖析(一):一个简单的Web服务器 

  2. Tomcat剖析(二):一个简单的Servlet服务器

 

这一节基于 《深度剖析Tomcat》第二章: 一个简单的Servlet服务器 总结而成。

上一节,我们了解了一个简单的Web服务器的总体处理流程是怎样的;这一节,我们开始搭建一个简单的Servlet容器,也就是增加实现了servlet的简单加载执行,而不仅仅是将文件内容输出到浏览器上。当然,这一节的servlet的实现是最简单的,用来了解整个Servlet的大致工作流程。

推荐先到我的github(地址在本文最后)上下载本书相关的代码,同时去上网下载这本书。

 

说到servlet,首先得先了解javax.servlet.Servlet接口,Servlet接口中的方法描述了servlet的生命周期方法,即init()、service()和destroy()。我们自己所有的servlet必须实现这个接口或者继承实现了这个接口的类(最一般的情况就是继承HttpServlet)。对于servlet生命周期的方法就不在这里具体描述了,可以自行查询关于“servlet生命周期”的资料。在这里只要知道Servlet接口可以完成“在请求第一次到来时初始化servlet、对每次请求调用service方法执行请求、servlet关闭时调用destroy方法销毁servlet”3个功能就够了。

 

一个Http请求过来时,Servlet服务器经历以下过程

1.Server创建一个serverSocket对象,等待Client发送请求

2.Client发送请求后,Server获取用户socket,从而得到请求的输入输出流

3.从输入输出流中创建request和response对象

4.解析request,如果是静态资源就创建StaticResourceProcessor实例,传递请求和响应给对应的方法,具体方法就是:从request中获取URI,判断文件是否存在,如果不存在就响应404 ,存在则将文件内容写到浏览器;如果是servlet请求就创建ServletProcessor1实例,同样传递请求和响应给对应的方法,完成加载servlet和调用Servlet的service方法执行。

5.关闭用户socket

6.判断URI是否为/SHUTDOWN,如果不是则重新进入等待请求状态,回到第2步,否则关闭服务器。

 

技术分享

 

整个流程主要包含5个类,HttpServer1、ServletProcessor1、StaticResourceProcessor、Request、Response。

HttpServer1.java:完成创建ServerSocket对象,获取Socket对象及其输入输出流,解析请求,创建Request对象和Response对象,并将不同类型的请求分派给不同的Processor处理

ServletProcessor1.java:当请求servlet时创建的对象,用于加载和执行servlet。

StaticResourceProcessor.java:当请求的资源是静态资源时创建的对象,调用Response对象的sendStaticResource()处理静态资源

Request.java:与上一节基本一样,是对输入流解析实现,获取URI。不同之处在于实现 javax.servlet.ServletRequest 接口,这一节中除了解析请求的方法外,其它未实现的方法置为默认值。

Reponse.java:与上一节基本一样,完成对浏览器的响应,包含对请求文件存在与不存在的处理。不同之处在于实现 javax.servlet.ServletResponse接口,这一节中除了getWriter方法外,其它未实现的方法置为默认值。

 

注意:在这一节中有些东西看起来是不合理的,以后的章节中会改进:

  1.对Servlet接口 仅仅是调用了service方法,没有调用int()和destroy()

  2.每一个servlet被请求时,servlet类就被加载一次

 

代码陈述更方便,以下开始将结合代码讲解

 

 HttpServer1.java

这个类和上一节HttpServer的非常相似。

不同之处只有在判断请求的类型时进行if else处理,而不是直接用上一节的response.sendStaticResource()直接将文件内容写到浏览器

对应的,如果判断确实是请求静态资源(即URL不以/servlet/开头)就调用StaticResourceProcessor处理器的process方法,显示文件到浏览器中

反之,如果判断是servlet,调用ServletProcessor1的process方法加载和执行servlet

 1 package ex02.pyrmont;
 2 
 3 import java.net.Socket;
 4 import java.net.ServerSocket;
 5 import java.net.InetAddress;
 6 import java.io.InputStream;
 7 import java.io.OutputStream;
 8 import java.io.IOException;
 9 
10 public class HttpServer1 {
11 
12     // 用于判断是否需要关闭服务器,默认是false
13     private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";
14     private boolean shutdown = false;
15 
16     public static void main(String[] args) {
17         HttpServer1 server = new HttpServer1();
18         server.await();
19     }
20 
21     public void await() {
22         
23         ServerSocket serverSocket = null;
24         int port = 8080;
25         try {
26             //创建服务器端的ServerSocket
27             serverSocket = new ServerSocket(port, 1,
28                     InetAddress.getByName("127.0.0.1"));
29         } catch (IOException e) {
30             e.printStackTrace();
31             System.exit(1);
32         }
33 
34         while (!shutdown) {
35             Socket socket = null;
36             InputStream input = null;
37             OutputStream output = null;
38             try {
39                 socket = serverSocket.accept();//Server一直等待直到Client发送请求
40                 input = socket.getInputStream();//接收请求后获取输入输出流
41                 output = socket.getOutputStream();
42 
43                 //创建request对象,传入input参数用于获取输入流的参数
44                 Request request = new Request(input);
45                 request.parse();//解析request对象,这一节同样也只是获取请求中请求行的URI
46 
47                 //创建response对象,传入output对象用于获取Writer对象
48                 Response response = new Response(output);
49                 response.setRequest(request);
50 
51                 //下面是这个类的关键所在
52                 //当URL是以/servlet/开头时说明请求servlet,使用ServletProcessor1处理器处理
53                 //否则说明是请求静态资源,由StaticResourceProcessor处理器处理
54                 if (request.getUri().startsWith("/servlet/")) {
55                     ServletProcessor1 processor = new ServletProcessor1();
56                     processor.process(request, response);
57                 } else {
58                     StaticResourceProcessor processor = new StaticResourceProcessor();
59                     processor.process(request, response);
60                 }
61 
62                 //关闭用户socket
63                 socket.close();
64                 //如果URI是/SHUTDOWN说明需要关闭服务器
65                 shutdown = request.getUri().equals(SHUTDOWN_COMMAND);
66             } catch (Exception e) {
67                 e.printStackTrace();
68                 System.exit(1);
69             }
70         }
71     }
72 }

HttpServer1.java的详细说明可以参照第一节的HttpServer类的详细说明

 由此类可以看出:

 要请求一个静态资源,就像是在第 1 节提到的,你可以在你的浏览器地址栏或者网址框里边敲入一个 URL:
  http://machineName:port/staticResource
为了请求一个 servlet,你可以使用下面的 URL:
  http://machineName:port/servlet/servletClass

 

 ServletProcessor1.java

 这个类是这一节的关键类,

仅仅只有一个process方法,但方法内部却没那么简单。

先上代码

 1 package ex02.pyrmont;
 2 
 3 import java.net.URL;
 4 import java.net.URLClassLoader;
 5 import java.net.URLStreamHandler;
 6 import java.io.File;
 7 import java.io.IOException;
 8 
 9 import javax.servlet.Servlet;
10 import javax.servlet.ServletRequest;
11 import javax.servlet.ServletResponse;
12 
13 public class ServletProcessor1 {
14 
15     public void process(Request request, Response response) {
16 
17         String uri = request.getUri();// 获取URI,如/servlet/className
18         // 要知道servlet名,截取第二段,即可获得className
19         String servletName = uri.substring(uri.lastIndexOf("/") + 1);
20         URLClassLoader loader = null;
21 
22         try {
23             //try块中用于创建URLClassLoader对象
24             URL[] urls = new URL[1];
25             URLStreamHandler streamHandler = null;
26             //Constants类存储静态常量,在本节最后贴上。
27             //Constants.WEB_ROOT即System.getProperty("user.dir") + File.separator  + "webroot";
28             File classPath = new File(Constants.WEB_ROOT);
29             String repository = (new URL("file", null,
30                     classPath.getCanonicalPath() + File.separator)).toString();
31             urls[0] = new URL(null, repository, streamHandler);
32             //经过以上过程后可以得到类似“file:E:/java/tomcat/servletName/”的路径
33             //最后URLClassLoader对象根据这个url获取到相应路径下serlvet的类加载器
34             loader = new URLClassLoader(urls);
35             
36         } catch (IOException e) {
37             System.out.println(e.toString());
38         }
39         Class myClass = null;
40         try {
41             myClass = loader.loadClass(servletName);  //根据反射获取Class对象
42         } catch (ClassNotFoundException e) {
43             System.out.println(e.toString());
44         }
45 
46         Servlet servlet = null;
47         try {
48             servlet = (Servlet) myClass.newInstance();//创建Servlet实例
49             servlet.service((ServletRequest) request,//执行servlet
50                     (ServletResponse) response);
51         } catch (Exception e) {
52             System.out.println(e.toString());
53         } catch (Throwable e) {
54             System.out.println(e.toString());
55         }
56 
57     }
58 }

ServletProcess1.java详细说明:
要加载 servlet,你可以使用 java.net.URLClassLoader 类,它是 java.lang.ClassLoader类的一个直接子类。

public URLClassLoader(URL[] urls); 这里 urls 是一个 java.net.URL 的对象数组,这些对象指向了加载类时候查找的位置。任何以/结尾的 URL 都假设是一个目录。

类加载器必须查找的地方只有一个,如工作目录下面的 webroot目录。因此,我们首先创建一个单个 URL 组成的数组。

URL 类提供了一系列的构造方法,所以有很多构造一个 URL 对象的方式。

一旦你拥有一个 URLClassLoader 实例,你使用它的 loadClass 方法去加载一个 servlet 类。(实在不懂怎么用可以自己查看API或者百度谷歌一下)

 然后,process 方法创建一个 servlet 类加载器的实例,  把它向下转换(downcast) 为
javax.servlet.Servlet, 并调用 servlet 的 service 方法

 

 StaticResourceProcessor.java

 这个类只有一个process方法,但确实很简单,如果理解了上一节,现在就没什么难理解了。

 1 package ex02.pyrmont;
 2 
 3 import java.io.IOException;
 4 
 5 public class StaticResourceProcessor {
 6 
 7     //存粹的打印文件内容到浏览器
 8     public void process(Request request, Response response) {
 9         try {
10             
11             response.sendStaticResource();
12         } catch (IOException e) {
13             e.printStackTrace();
14         }
15     }
16 }

 

 Request.java和Response.java内容很多而且不是本节关键代码,大家可以根据上面这两个类的简单介绍

并结合我github上的Tomcat4的代码找到ex02.pyrmont包下的这两个类,

最后与上一节这两个类的代码(对应ex01.pyrmont包)对比以下。应该就没什么大问题,这里就不说了

 

下面的折叠代码是Constants.java,打消大家的疑虑

技术分享
package ex02.pyrmont;

import java.io.File;

public class Constants {
  public static final String WEB_ROOT =
    System.getProperty("user.dir") + File.separator  + "webroot";
}
View Code

 

 

最后说说上面搭建的代码的安全性问题

 在 ServletProcessor1 类的 process 方法,你向上转换ex02.pyrmont.Request 实例为 javax.servlet.ServletRequest ,并作为第一个参数传递给servlet 的 service 方 法 。 你 也 向 下 转 换 ex02.pyrmont.Response 实 例 为javax.servlet.ServletResponse,并作为第二个参数传递给 servlet 的 service 方法。

try {
servlet = (Servlet) myClass.newInstance();
servlet.service((ServletRequest) request,(ServletResponse) response);
}


这会危害安全性。知道这个 servlet 容器的内部运作的 Servlet 程序员可以分别把ServletRequest 和 ServletResponse 实 例 向 下 转 换 为 ex02.pyrmont.Request 和
ex02.pyrmont.Response,并调用他们的公共方法。拥有一个 Request 实例,它们就可以调用 parse方法。拥有一个 Response 实例,就可以调用 sendStaticResource 方法。

 为了解决这个问题,我们增加了两个 façade 类: RequestFacade 和 ResponseFacade。RequestFacade 实现了 ServletRequest 接口并通过在构造方法中传递一个引用了
ServletRequest 对象的 Request 实例作为参数来实例化。ServletRequest 接口中每个方法的实现都调用了 Request 对象的相应方法。然而 ServletRequest 对象本身是私有的,并不能在类的外部访问。我们构造了一个 RequestFacade 对象并把它传递给 service 方法,而不是向下转换Request 对象为 ServletRequest 对象并传递给 service 方法。 Servlet 程序员仍然可以向下转换ServletRequest 实例为 RequestFacade,不过它们只可以访问 ServletRequest 接口里边的公共方法。现在 parseUri 方法就是安全的了。

 

其实我也不想放上这两个类,但是还是折叠给上吧。

 RequestFacade.java

技术分享
  1 package ex02.pyrmont;
  2 
  3 import java.io.IOException;
  4 import java.io.BufferedReader;
  5 import java.io.UnsupportedEncodingException;
  6 import java.util.Enumeration;
  7 import java.util.Locale;
  8 import java.util.Map;
  9 
 10 import javax.servlet.AsyncContext;
 11 import javax.servlet.DispatcherType;
 12 import javax.servlet.RequestDispatcher;
 13 import javax.servlet.ServletContext;
 14 import javax.servlet.ServletInputStream;
 15 import javax.servlet.ServletRequest;
 16 import javax.servlet.ServletResponse;
 17 
 18 public class RequestFacade implements ServletRequest {
 19 
 20     private ServletRequest request = null;
 21 
 22     public RequestFacade(Request request) {
 23         this.request = request;
 24     }
 25 
 26     /* implementation of the ServletRequest */
 27     public Object getAttribute(String attribute) {
 28         return request.getAttribute(attribute);
 29     }
 30 
 31     public Enumeration getAttributeNames() {
 32         return request.getAttributeNames();
 33     }
 34 
 35     public String getRealPath(String path) {
 36         return request.getRealPath(path);
 37     }
 38 
 39     public RequestDispatcher getRequestDispatcher(String path) {
 40         return request.getRequestDispatcher(path);
 41     }
 42 
 43     public boolean isSecure() {
 44         return request.isSecure();
 45     }
 46 
 47     public String getCharacterEncoding() {
 48         return request.getCharacterEncoding();
 49     }
 50 
 51     public int getContentLength() {
 52         return request.getContentLength();
 53     }
 54 
 55     public String getContentType() {
 56         return request.getContentType();
 57     }
 58 
 59     public ServletInputStream getInputStream() throws IOException {
 60         return request.getInputStream();
 61     }
 62 
 63     public Locale getLocale() {
 64         return request.getLocale();
 65     }
 66 
 67     public Enumeration getLocales() {
 68         return request.getLocales();
 69     }
 70 
 71     public String getParameter(String name) {
 72         return request.getParameter(name);
 73     }
 74 
 75     public Map getParameterMap() {
 76         return request.getParameterMap();
 77     }
 78 
 79     public Enumeration getParameterNames() {
 80         return request.getParameterNames();
 81     }
 82 
 83     public String[] getParameterValues(String parameter) {
 84         return request.getParameterValues(parameter);
 85     }
 86 
 87     public String getProtocol() {
 88         return request.getProtocol();
 89     }
 90 
 91     public BufferedReader getReader() throws IOException {
 92         return request.getReader();
 93     }
 94 
 95     public String getRemoteAddr() {
 96         return request.getRemoteAddr();
 97     }
 98 
 99     public String getRemoteHost() {
100         return request.getRemoteHost();
101     }
102 
103     public String getScheme() {
104         return request.getScheme();
105     }
106 
107     public String getServerName() {
108         return request.getServerName();
109     }
110 
111     public int getServerPort() {
112         return request.getServerPort();
113     }
114 
115     public void removeAttribute(String attribute) {
116         request.removeAttribute(attribute);
117     }
118 
119     public void setAttribute(String key, Object value) {
120         request.setAttribute(key, value);
121     }
122 
123     public void setCharacterEncoding(String encoding)
124             throws UnsupportedEncodingException {
125         request.setCharacterEncoding(encoding);
126     }
127 
128     @Override
129     public int getRemotePort() {
130         // TODO Auto-generated method stub
131         return 0;
132     }
133 
134     @Override
135     public String getLocalName() {
136         // TODO Auto-generated method stub
137         return null;
138     }
139 
140     @Override
141     public String getLocalAddr() {
142         // TODO Auto-generated method stub
143         return null;
144     }
145 
146     @Override
147     public int getLocalPort() {
148         // TODO Auto-generated method stub
149         return 0;
150     }
151 
152     @Override
153     public ServletContext getServletContext() {
154         // TODO Auto-generated method stub
155         return null;
156     }
157 
158     @Override
159     public AsyncContext startAsync() {
160         // TODO Auto-generated method stub
161         return null;
162     }
163 
164     @Override
165     public AsyncContext startAsync(ServletRequest servletrequest,
166             ServletResponse servletresponse) {
167         // TODO Auto-generated method stub
168         return null;
169     }
170 
171     @Override
172     public boolean isAsyncStarted() {
173         // TODO Auto-generated method stub
174         return false;
175     }
176 
177     @Override
178     public boolean isAsyncSupported() {
179         // TODO Auto-generated method stub
180         return false;
181     }
182 
183     @Override
184     public AsyncContext getAsyncContext() {
185         // TODO Auto-generated method stub
186         return null;
187     }
188 
189     @Override
190     public DispatcherType getDispatcherType() {
191         // TODO Auto-generated method stub
192         return null;
193     }
194 
195 }
View Code

ResponseFacade.java

技术分享
 1 package ex02.pyrmont;
 2 
 3 import java.io.IOException;
 4 import java.io.PrintWriter;
 5 import java.util.Locale;
 6 
 7 import javax.servlet.ServletResponse;
 8 import javax.servlet.ServletOutputStream;
 9 
10 public class ResponseFacade implements ServletResponse {
11 
12     private ServletResponse response;
13 
14     public ResponseFacade(Response response) {
15         this.response = response;
16     }
17 
18     public void flushBuffer() throws IOException {
19         response.flushBuffer();
20     }
21 
22     public int getBufferSize() {
23         return response.getBufferSize();
24     }
25 
26     public String getCharacterEncoding() {
27         return response.getCharacterEncoding();
28     }
29 
30     public Locale getLocale() {
31         return response.getLocale();
32     }
33 
34     public ServletOutputStream getOutputStream() throws IOException {
35         return response.getOutputStream();
36     }
37 
38     public PrintWriter getWriter() throws IOException {
39         return response.getWriter();
40     }
41 
42     public boolean isCommitted() {
43         return response.isCommitted();
44     }
45 
46     public void reset() {
47         response.reset();
48     }
49 
50     public void resetBuffer() {
51         response.resetBuffer();
52     }
53 
54     public void setBufferSize(int size) {
55         response.setBufferSize(size);
56     }
57 
58     public void setContentLength(int length) {
59         response.setContentLength(length);
60     }
61 
62     public void setContentType(String type) {
63         response.setContentType(type);
64     }
65 
66     public void setLocale(Locale locale) {
67         response.setLocale(locale);
68     }
69 
70     @Override
71     public String getContentType() {
72         // TODO Auto-generated method stub
73         return null;
74     }
75 
76     @Override
77     public void setCharacterEncoding(String s) {
78         // TODO Auto-generated method stub
79 
80     }
81 
82 }
View Code

 

最后根据上面的描述,ServletProcess1.java改成ServletProcess2.java,使用RequestFacade和ResponseFacade进行封装

改变的内容不多,如下

1  Servlet servlet = null;
2     RequestFacade requestFacade = new RequestFacade(request);
3     ResponseFacade responseFacade = new ResponseFacade(response);
4     try {
5       servlet = (Servlet) myClass.newInstance();
6       servlet.service((ServletRequest) requestFacade, (ServletResponse) responseFacade);
7     }

当然,HttpServer1.java改为HttpServer2.java,改变之处只有一个地方,即创建ServletProcess1实例改成ServletProcess2实例

 

这一节就到这里,我们完成对servlet请求和静态资源请求分别处理的简单实现,相比上一节难度大了一些。下一节将开始Tomcat的连接器。

如果有什么疑问或错误,可以发表评论或者加我QQ:1096101803告知,谢谢。

 

附:

相应代码可以在我的github上找到下载,拷贝到eclipse,然后打开对应包的代码即可。

https://github.com/zebinlin/Tomcat4-src

如发现编译错误,可能是由于jdk不同版本对编译的要求不同导致的,可以不管,供学习研究使用。

 

 

 

 

 

Tomcat剖析(二)

标签:

原文地址:http://www.cnblogs.com/lzb1096101803/p/4785246.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!