上一节(How Tomcat Works 1 编写一个简单静态web服务器)编写了一个简单的web服务器,只能处理静态的资源,本节将继续向前迈出一个小步,创建两个不同的servlet容器,能够利用servlet简单的处理动态内容。注意每节的代码都是基于上一节的继续丰富,因此有必要从第一节开始看起。
在编写代码之前,需要先大体了解一下Servlet是什么,方便后面的理解,下面就是一个最简单的Servlet什么也没做:
package prymont; import java.io.IOException; import javax.servlet.Servlet; import javax.servlet.ServletConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; /** * 每个Servlet都实现Servlet接口 * 该接口有5个方法需要实现,该Servlet只是为了让大家对 * Servlet有个整体的印象,因此所有方法都未实现。 */ public class PrimitiveServlet implements Servlet{ //当servlet容器正在被关闭或者servlet容器内存不够的时候,该方法由servlet容器调用,且只调用一次 //该方法通常用来清除资源。 @Override public void destroy() { } @Override public ServletConfig getServletConfig() { return null; } @Override public String getServletInfo() { return null; } //当servlet已经初始化的时候,该方法由servlet容器调用,且只调用一次 //该方法适合做一些一次性的加载动作,比如数据库驱动等。 @Override public void init(ServletConfig arg0) throws ServletException { } //servlet容器负责为每一次请求调用一次service方法,并且传递一个ServletRequest(封装客户端请求) //和ServletResponse(封装响应)对象。 @Override public void service(ServletRequest arg0, ServletResponse arg1) throws ServletException, IOException { } }
先来说说我们的第一个版本的Servlet容器要实现的功能:等待Http请求,如果是请求静态资源则交给静态资源处理器,如果是请求动态资源则加载相应的Servlet并调用它的service方法,同时传递ServletRequest和ServletResponse对象。(第一个版本每次请求servlet类都被加载)
本版本针对上次主要新增StaticResourceProcessor和ServletProcessor1两个类分别处理静态资源和动态资源,同时Request和Response对象也各自集成了
http://machineName:port/staticResource请求的是一个静态资源,http://localhost:8080/servlet/PrimitiveServlet请求的则是一个动态的资源,因此对于HttpServer1只需要稍稍改动一下即可满足需求。
StaticResourceProcessor只是简单的调用了response的sendStaticResource()方法,也没有可讲的,下面重点讲解一下ServletProcessor1的实现:
package server1; import java.io.File; import java.io.IOException; import java.net.MalformedURLException; import java.net.URL; import java.net.URLClassLoader; import java.net.URLStreamHandler; import javax.servlet.Servlet; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; /** * 完成servlet类的动态加载 * 并调用servlet的service的方法 * */ public class ServletProcessor1 { /** * 加载Servlet * * @param request * @param response */ public void process(Request request, Response response) { // /servlet/servetclass String uri = request.getUri(); // 截取servlet名字 String servletName = uri.substring(uri.lastIndexOf("/") + 1); // jdk提供的URL类加载器,根据URL加载class URLClassLoader loader = null; URL[] urls = new URL[1]; // 指定加载webapp下所有的class文件 File classPath = new File(Constant.WEB_APP); URLStreamHandler streamHandler = null; try { // 使用file协议从本机的classPath加载class // getCanonicalPath返回绝对路径(不带.) String repository = new URL("file", null, classPath.getCanonicalPath() + File.separator).toString(); urls[0] = new URL(null, repository, streamHandler); } catch (MalformedURLException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } loader = new URLClassLoader(urls); Class clazz = null; try { // 根据servlet名字加载class,这里对名字的处理并不完善,实际需要根据包名做拼接 //为了简单servlet类全不不带包名 clazz = loader.loadClass(servletName); } catch (ClassNotFoundException e) { e.printStackTrace(); } Servlet servlet = null; try { // 反射创建实例 servlet = (Servlet) clazz.newInstance(); // 向下转型调用service方法 servlet.service((ServletRequest) request, (ServletResponse) response); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (ServletException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } }上面就是处理Servlet的整个过程,可以通过http://localhost:8080/servlet/PrimitiveServlet来访问我们的servlet。
我们的第一个程序有一个严重的问题,就是当传递给servlet的service方法的时候将Request和Response对象都向上转型了,这样如果知道内部实现的人,则可以在写自己的Servlet的时候向下转型成Request和Response对象,并调用他们的方法,实际上这两个对象是容器私有的不应该暴露给开发者(具体是有些方法不能让开发者使用),有一个方法是让这两个类使用默认的包访问权限,其实有一个更优雅的实现方式就是门面模式。
在本节的第二版中增加两个类,RequestFacade和ResponseFacade用来控制某些方法的可见性。具体的方式是给RequestFacade提供一个带参构造函数,参数类型为Request对象,门面类封装servletrequest方法,其实只是传递给Request对象实现,这样在调用service方法时传递的facade对象,这样即使通过向下转型得到的也是被封装过的Facade对象。RequestFacade对象片段如下:
public class RequestFacade implements ServletRequest { private ServletRequest request = null; public RequestFacade(Request request) { this.request = request; } /* implementation of the ServletRequest*/ public Object getAttribute(String attribute) { return request.getAttribute(attribute); }ResponseFacade对象类似,不贴出来代码了。
还有一点不同的是ServletProcessor1的部分处理,具体变化的如下:
Servlet servlet = null; RequestFacade requestFacade = new RequestFacade(request); ResponseFacade responseFacade = new ResponseFacade(response); try { // 反射创建实例 servlet = (Servlet) clazz.newInstance(); // 向下转型调用service方法 servlet.service((ServletRequest) requestFacade, (ServletResponse) responseFacade); } catch (InstantiationException e) { e.printStackTrace(); } catch (IllegalAccessException e) { e.printStackTrace(); } catch (ServletException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); }通过Request对象构造RequestFacade对象并向上转型传递给service方法。
以上就是本节实现的一个简单的servlet容器。下一节将会接触到tomcat中一个非常重要的概念——连接器
原文地址:http://blog.csdn.net/tangyongzhe/article/details/9004165