本文主要是介绍How Tomcat Works 2,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
上一节(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容器调用,且只调用一次//该方法通常用来清除资源。@Overridepublic void destroy() {}@Overridepublic ServletConfig getServletConfig() {return null;}@Overridepublic String getServletInfo() {return null;}//当servlet已经初始化的时候,该方法由servlet容器调用,且只调用一次//该方法适合做一些一次性的加载动作,比如数据库驱动等。@Overridepublic void init(ServletConfig arg0) throws ServletException {}//servlet容器负责为每一次请求调用一次service方法,并且传递一个ServletRequest(封装客户端请求)//和ServletResponse(封装响应)对象。@Overridepublic 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/servetclassString uri = request.getUri();// 截取servlet名字String servletName = uri.substring(uri.lastIndexOf("/") + 1);// jdk提供的URL类加载器,根据URL加载classURLClassLoader 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中一个非常重要的概念——连接器
这篇关于How Tomcat Works 2的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!