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方法。其他请求都是直接调用相应方法。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏

Cassandra Java 使用TimeUUIDType

参考地址 http://wiki.apache.org/cassandra/FAQ#working_with_timeuuid_in_java

2153
来自专栏小灰灰

JavaWeb三大组件之Servlet学习

JavaWeb三大组件之Servlet学习 平时直接用springmvc较多,都没怎么接触底层的Servlet,导致对一些基本的知识点了解都不够,所以今天专门...

2339
来自专栏安恒网络空间安全讲武堂

nox&amp;CSAW部分pwn题解

暑假的时候遇到了一群一起学习安全的小伙伴,在他们的诱劝下,开始接触国外的CTF比赛,作为最菜的pwn选手就试着先打两场比赛试试水,结果发现国外比赛真有意思哎嘿。

1613
来自专栏好好学java的技术栈

HttpServletRequest这篇文章就够了

HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,通过这个对象提供的...

2K0
来自专栏猿天地

用aop加redis实现通用接口缓存

系统在高并发场景下,最有用的三个方法是缓存,限流,降级。 缓存就是其中之一,目前缓存基本上是用redis或者memcached。 redis和memcached...

3287
来自专栏IT可乐

Servlet 详解

1、什么是 Servlet? Java Servlet 是运行在 Web 服务器或应用服务器上的程序,它是作为来自 Web 浏览器或其他 HTTP 客户端的请...

2227
来自专栏别先生

Ajax和JSON完成二级菜单联动的功能

首先需要找好JSON的包哦:   链接:http://pan.baidu.com/s/1jH6gN46 密码:lbh1 1:首先创建一个前台页面,比如seco...

24610
来自专栏好好学java的技术栈

一文读懂HttpServletRequest

HttpServletRequest对象代表客户端的请求,当客户端通过HTTP协议访问服务器时,HTTP请求头中的所有信息都封装在这个对象中,通过这个对象提供的...

1221
来自专栏一个番茄说

让你在WebView中用JS调Native Object

之所做这个东西,源于之前项目中需要把一些页面用webView来呈现,但是web中需要调用native的方法,比如获取本地存的某些数据、调用摄像头等等,这里也就是...

1003
来自专栏分布式系统进阶

KafkaController分析3-ControllerChannelManager

使用NetworkClient连接到broker node, 使用selector处理网络IO;

731

扫码关注云+社区