HttpServlet源码分析

1.HttpServlet的用法

提供了创建Http Servlet的抽象类,通过实现此类定义自己的Servlet

2.HttpServlet是否是线程安全

先说结论:HttpServlet不是线程安全

tomcat在只在第一次有请求的时候加载Servlet,加载之后调用init方法进行初始化,容器中只会保存一个Servlet对象,当有Http请求的时候会调用service方法对请求进行处理。所以Servlet不是线程安全的,当我们在Servlet中定义类变量或者处理共享资源时,要注意线程安全问题。简单的理解就是:单例多线程。
以下是对Servlet的测试代码:
private static int staticCount = 0;

 private int count  = 0;

 @Override
 protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
     PrintWriter out = resp.getWriter();
     out.println("count = " + count++ + "staticCount = " + staticCount++);
 }

 @Override
 public void init() throws ServletException {
     super.init();
     System.out.println("init servlet!");
 }

运行tomcat,在浏览器中第一次发送http请求的时候控制台会输出

  init servlet!

浏览器中多次发送请求后会输出:

count = 14staticCount = 14

由此验证以上结论:单例多线程,Servlet不是线程安全的。

3.关键方法分析:service()

当接收到Http请求的时候会有一个public的service方法,此方法中只是对参数类型进行判断和强制转换,然后调用此protected方法处理:

protected void service(HttpServletRequest req, HttpServletResponse resp)
  throws ServletException, IOException
{
  String method = req.getMethod();

  if (method.equals(METHOD_GET)) {
      long lastModified = getLastModified(req);
      if (lastModified == -1) {
          // servlet doesn't support if-modified-since, no reason
          // to go through further expensive logic
          doGet(req, resp);
      } else {
          long ifModifiedSince = req.getDateHeader(HEADER_IFMODSINCE);
          if (ifModifiedSince < lastModified) {
              // If the servlet mod time is later, call doGet()
              // Round down to the nearest second for a proper compare
              // A ifModifiedSince of -1 will always be less
              maybeSetLastModified(resp, lastModified);
              doGet(req, resp);
          } else {
              resp.setStatus(HttpServletResponse.SC_NOT_MODIFIED);
          }
      }

  } else if (method.equals(METHOD_HEAD)) {
      long lastModified = getLastModified(req);
      maybeSetLastModified(resp, lastModified);
      doHead(req, resp);

  } else if (method.equals(METHOD_POST)) {
      doPost(req, resp);

  } else if (method.equals(METHOD_PUT)) {
      doPut(req, resp);

  } else if (method.equals(METHOD_DELETE)) {
      doDelete(req, resp);

  } else if (method.equals(METHOD_OPTIONS)) {
      doOptions(req,resp);

  } else if (method.equals(METHOD_TRACE)) {
      doTrace(req,resp);

  } else {
      //
      // Note that this means NO servlet supports whatever
      // method was requested, anywhere on this server.
      //

      String errMsg = lStrings.getString("http.method_not_implemented");
      Object[] errArgs = new Object[1];
      errArgs[0] = method;
      errMsg = MessageFormat.format(errMsg, errArgs);

      resp.sendError(HttpServletResponse.SC_NOT_IMPLEMENTED, errMsg);
  }
}

此方法中比较关键的地方是对get请求的处理:在处理get请求的时候会对所请求中的资源上次修改时间和实际修改时间比较,根据比较结果返回状态码或者继续请求资源。此方式需要Servlet实现getLastModified方法。其他请求都是直接调用相应方法。

posted @ 2017-08-24 22:08  代码改变世界-Coding  阅读(1045)  评论(0编辑  收藏  举报