Servlet与模板模式那些事

Java的开源给了我们更多机会,去了解组件内部的实现。从事Java Web开发,大家都会跟直接或间接的跟Servlet打交道。今天我们就来谈谈Servlet。

我们在手写Servlet时,会继承HttpServlet类。在HttpServlet类中,有几个重要的方法:doPost()方法、doGet()方法、service()方法。当我们调用自己写的Servlet时,系统会默认调用doGet()方法;或者说,当我们指定一种传输方式,系统会自动调用我们重写的方法。这是采用设计模式中模板模式实现的。

Servlet是由Tomcat之类的servlet容器来调用处理浏览器请求的,并需要集成基类HttpServlet。反编译出HttpServlet的class文件,我们可以看出,该类中存在两个service()方法:

1、
  protected void service(HttpServletRequest req, HttpServletResponse resp)
    throws ServletException, IOException{……}

2、
  public void service(ServletRequest req, ServletResponse res)
    throws ServletException, IOException{……}

public修饰的service方法,其实是重写了父类GenericServlet中的service()方法,它主要完成的功能是接收客户端请求,并将resquest和response传递给service()方法。父类GenericServlet中,则是service()方法的最高级抽象,给出的是service()抽象方法。


  public abstract void service(ServletRequest paramServletRequest, ServletResponse paramServletResponse)
    throws ServletException, IOException;

而protected修饰的service方法,Tomcat容器实际调用的是这个service()方法。service()方法的实现就是根据http求情的类型(get、post),将处理委派给doGet、doPost等方法,由这些子类的方法来最终处理浏览器的求情。

由此可知:HttpServlet定义了一个处理的框架,或者说是模板,实现Servlet只需要继承HttpServlet并重写doPost()和doGet()方法就可以了。

下面看看模版模式的实现:

//父类HttpServlet
public abstract class HttpServlet extends GenericServlet
  implements Serializable
{
  protected void service(HttpServletRequest req, HttpServletResponse resp)
    throws ServletException, IOException
  {……}

 public void service(ServletRequest req, ServletResponse res)
    throws ServletException, IOException
  {
    HttpServletRequest request;
    HttpServletResponse response;
    try
    {
      request = (HttpServletRequest)req;
      response = (HttpServletResponse)res;
    } catch (ClassCastException e) {
      throw new ServletException("non-HTTP request or response");
    }
    //调用protected service
    service(request, response);
  }
}

//自己写的Servlet,继承HttpServlet
import java.io.IOException;
import java.io.PrintStream;
import java.io.PrintWriter;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class LoginServlet extends HttpServlet
{
	//重写父类中的doGet方法
  public void doGet(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException
  {
    String strOne = request.getParameter("username");
    String strTwo = request.getParameter("password");

    System.out.println("username=" + strOne);
    System.out.println("password=" + strTwo);

    response.setContentType("text/html");
    response.getWriter().println("Login Success!!");
  }
  //重写父类中的doPost方法
   public void doPost(HttpServletRequest request, HttpServletResponse response)
    throws ServletException, IOException
    {
    	doGet(request,response);
  	}
}

总结,这种实现,重复的代码都上升到了父类中去,而变化的部分在子类中具体实现。

你可能感兴趣的:(java,object,语言)