JavaWeb-Servlet笔记

JavaWeb-Servlet笔记

1、什么是Servlet?

Servlet(Server Applet)是Java Servlet的简称,称为小服务程序或服务连接器,用Java编写的服务器端程序,具有独立于平台和协议的特性,主要功能在于交互式地浏览和生成数据,生成动态Web内容

狭义的Servlet是指Java语言实现的一个接口,广义的Servlet是指任何实现了这个Servlet接口的类,一般情况下,人们将Servlet理解为后者。Servlet运行于支持Java的应用服务器中。从原理上讲,Servlet可以响应任何类型的请求,但绝大多数情况下Servlet只用来扩展基于HTTP协议的Web服务器。

2、Servlet快速入门

  1. 创建一个maven项目

JavaWeb-Servlet笔记
2. 导入Servlet的依赖

   <dependencies>
       <dependency>
           <groupId>javax.servlet</groupId>
           <artifactId>servlet-api</artifactId>
           <version>2.5</version>
       </dependency>
   </dependencies>

3.定义一个类,实现Servlet接口

public class ServletTest01 implements Servlet

4.实现接口中的抽象方法

import javax.servlet.*;
import java.io.IOException;

public class ServletTest01 implements Servlet {
    @Override
    public void init(ServletConfig config) throws ServletException {
        
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {
        System.out.println("Hello Servlet!");
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {

    }
}

5.配置Servlet

在web.xml中配置:

  <!--servlet描述信息-->
<servlet> 
        <!--定义当前servlet的名字,自定义-->
        <servlet-name>HelloServlet</servlet-name>
        <!--带有包名的全限定类名-->
        <servlet-class>com.wang.servlet.ServletTest01</servlet-class>
    </servlet>

    <!--任何一个servlet都对应一个servlet-mapping-->
    <servlet-mapping>
        <!--和上面的servlet名字要一致-->
        <servlet-name>HelloServlet</servlet-name>
        <!--当前servlet的一个路径,必须以"/"开头,路径名自定义-->
        <url-pattern>/helloServlet</url-pattern>
    </servlet-mapping>

这里要注意,使用模板创建maven项目的web.xml文件格式是这样的,格式不对且版本较低

JavaWeb-Servlet笔记
建议直接创建一个空的项目,创建后在项目名处右键添加web支持,可以看到版本是4.0

JavaWeb-Servlet笔记
JavaWeb-Servlet笔记
这个项目里的web.xml文件是这样的,这个样子的web.xml是我们经常使用的
JavaWeb-Servlet笔记

<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0">

</web-app>

6.启动Tomcat
JavaWeb-Servlet笔记
JavaWeb-Servlet笔记
JavaWeb-Servlet笔记
JavaWeb-Servlet笔记
JavaWeb-Servlet笔记
配置成功后启动Tomcat
JavaWeb-Servlet笔记
7. 打开浏览器,输入http://localhost:8080/helloServlet,回车,在控制台输出结果,入门成功

JavaWeb-Servlet笔记

3、Servlet执行原理

JavaWeb-Servlet笔记

4、Servlet的生命周期

  • 什么是Servlet对象生命周期?

    一个Servlet对象从出生到最后的死亡,经历了怎么样的过程

  • Servlet对象是由谁来维护的?

    • Servlet对象的创建,对象中方法的调用,对象最终的销毁,程序员是无权干预的
    • Servlet对象的生命周期是有Tomcat服务器(WEB Server)全权负责的
    • Tomcat服务器通常又称为WEB容器(WEB Container),Servlet对象的死活由WEB容器来管理
    • web容器底层应该有一个HashMap集合,在这个集合中存储了Servlet对象(WEB容器创建的)和请求路径之间的关系,而我们自己new的Servlet对象不在这个容器当中,所以不会被WEB容器管理
  • 研究:

    • 默认情况下,服务器在启动的时候Servlet对象有没有被创建出来?

      • 在Servlet中提供一个无参数的构造方法,启动服务器的时候看看构造方法是否执行

JavaWeb-Servlet笔记
- 经过测试可以知道,Servlet的无参构造方法没有执行,从而得出结论,默认情况下,服务器在启动的时候Servlet对象并不会被实例化

  • 这个设计是合理的。用户没有发送请求之前,如果提前创建出来所有的Servlet对象,必然是耗费内存的,并且创建出来的Servlet如果一直没有用户访问,显然这个Servlet对象是没有必要创建的

  • 怎么让服务器启动的时候创建Servlet对象呢?

    • 在Servlet标签中添加子标签,在该子标签中填写整数,越小的整数优先级越高

        <servlet>
              <servlet-name>HelloServlet</servlet-name>
              <servlet-class>com.wang.servlet.ServletTest01</servlet-class>
              <load-on-startup>1</load-on-startup>
          </servlet>
          <servlet-mapping>
              <servlet-name>HelloServlet</servlet-name>
              <url-pattern>/helloServlet</url-pattern>
          </servlet-mapping>
      

    启动Tomcat后,控制台“输出了ServletTest01的无参构造方法执行了”,说明这是Servlet对象已经被实例化了
    JavaWeb-Servlet笔记

  • Servlet对象生命周期

    • 默认情况下服务器启动的时候ServletTest01对象并没有被实例化

    • 用户发送第一次请求的时候,控制台输出了以下内容

JavaWeb-Servlet笔记

  • 根据以上输出内容得出结论

    • 用户在发送第一次请求的时候Servlet对象被实例化(ServletTest01的构造方法被执行了,并且执行的是无参数构造方法)
    • ServletTest01对象被创建出来之后,Tomcat服务器马上调用了ServletTest01对象的init方法,(init方法在执行的时候,ServletTest01对象已经存在了,已经被创建出来了)
    • 用户发送第一从请求的时候,init方法执行之后,Tomcat服务器马上调用ServletTest01对象的service方法
  • 用户继续发送第二次请求,控制台输出了以下内容

JavaWeb-Servlet笔记

  • 根据以上输出结果得知,用户在发送第二次,或者第三次,第n次请求的时候,Servlet对象并没有新建,还是使用之前创建好的Servlet对象,直接调用Servlet对象的service方法,这说明

    • 第一:Servlet对象是单例的(单实例的,但是要注意:Servlet对象是单实例的,但是Servlet类并不符合单例模式,我们称之为假单例。之所以单例是因为Servlet对象的创建程序员管不着,这个对象的创建只能是Tomcat来说了算,Tomcat只创建一个,所以导致了单例,但是属于假单例。真单例模式,构造方法是私有化的)
    • 第二:无参数构造方法,init方法只在第一次用户发送请求的时候执行,也就是说无参数构造方法只执行一次,init方法也只被Tomcat服务器调用一次
    • 第三:只要用户发送一次请求:service方法必然会被Tomcat服务器调用一次,发送100次请求,service方法会被调用100次
  • 关闭服务器的时候,控制台输出了以下内容
    JavaWeb-Servlet笔记

  • 通过以上输出内容,可以得出以下结论

    • Servlet的destroy方法只被Tomcat服务器调用一次
    • destroy方法是在什么时候被调用的?
      • 在服务器关闭的时候
      • 因为服务器关闭的时候要销毁ServletTest01对象的内存
      • 服务器在销毁ServletTest01对象内存之前,Tomcat服务器会自动调用ServletTest01的destroy方法
      • 就好比,在离开家之前,要先检查门窗是否关好一样,需要先检查好了再离开,即执行destroy方法结束之后,ServletTest01对象的内存才会被Tomcat释放
  • Servlet对象更像一个人的一生

    • Servlet的无参构造方法执行,标志着你出生了
    • Servlet对象的init方法的执行,标志着你正在接受教育
    • Servlet对象的service方法的执行,标志着你已经开始工作了,已经开始为人类提供服务了
    • Servlet对象的destroy方法的执行,标志着临终,要交代遗言
  • 关于Servlet类中方法的调用次数?

    • 构造方法只执行一次
    • init方法只执行一次
    • service方法:用户发送一次请求则执行一次,发送N次请求则执行N次
    • destroy方法只执行一次
  • 当我们Servlet类中编写一个有参数的构造方法,如果没有手动编写无参数构造方法会出现什么问题?
    JavaWeb-Servlet笔记

    • 报错了,500错误
    • 500是一个HTTP协议的错误状态码
    • 500一般情况下是因为服务器端的java程序出现了异常(服务器端的错误都是500错误:服务器内部错误)
    • 如果没有无参数的构造方法,会导致出现500错误,无法实例化Servlet对象
    • 所以,一定要注意,在Servlet开发当中,不建议程序员来定义构造方法,因为定义不当,一不小心就会导致无法实例化Servlet对象
  • 思考:Servlet的无参数构造方法是在对象第一次创建的时候执行,并且只执行一次,init方法也是在对象第一次创建的时候执行,并且也只执行一次,那么这个无参数构造方法可以代替调init方法吗?

    • 不能
    • Servlet规范中有要求,作为JavaWeb程序员,编写Servlet类的时候,不建议手动编写构造方法,因为编写构造方法,很容易让无参数构造方法小时,这个操作可能会导致Servlet对象无法实例化,所以init方法是有存在的必要的
  • init、service、destroy方法中使用最多的是哪个方法?

    • service方法,service方法是一定要实现的,因为service方法是处理用户请求的核心方法
    • init方法很少用,通常在init方法当中做初始化操作,并且这个初始化只需要执行一次,例如:初始化数据库连接池,初始化线程池…
    • destroy方法很少用,通常在destroy方法中进行资源的关闭,马上对象就要被销毁了,还有什么没有关闭的,抓紧时间关闭资源,还有什么资源没保存的,抓紧时间保存一下

5、Servlet的体系结构

在上面的ServletTest01类中,我们实现了Servlet接口,实际只是需要service方法,其他方法大多数情况下是用不到的,那么有什么方法能解决这个问题呢?

我们来看看Servlet的体系结构

  • 查看API可知,Servlet接口有两个实现类
    JavaWeb-Servlet笔记
  • 其中,GenericServlet是实现Servlet 接口的抽象类,源码如下
package javax.servlet;

import java.io.IOException;
import java.util.Enumeration;
import java.util.ResourceBundle;

public abstract class GenericServlet 
    implements Servlet, ServletConfig, java.io.Serializable
{
    private static final String LSTRING_FILE = "javax.servlet.LocalStrings";
    private static ResourceBundle lStrings =
        ResourceBundle.getBundle(LSTRING_FILE);

    private transient ServletConfig config;
    
    public GenericServlet() { }
    
    public void destroy() {
    }
    
    public String getInitParameter(String name) {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getInitParameter(name);
    }
    
    public Enumeration getInitParameterNames() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getInitParameterNames();
    }   
    
    public ServletConfig getServletConfig() {
	return config;
    }
    
    public ServletContext getServletContext() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getServletContext();
    }

    public String getServletInfo() {
	return "";
    }

    public void init(ServletConfig config) throws ServletException {
	this.config = config;
	this.init();
    }

    public void init() throws ServletException {

    }
     
    public void log(String msg) {
	getServletContext().log(getServletName() + ": "+ msg);
    }

    public void log(String message, Throwable t) {
	getServletContext().log(getServletName() + ": " + message, t);
    }
    
    public abstract void service(ServletRequest req, ServletResponse res)
	throws ServletException, IOException;

    public String getServletName() {
        ServletConfig sc = getServletConfig();
        if (sc == null) {
            throw new IllegalStateException(
                lStrings.getString("err.servlet_config_not_initialized"));
        }

        return sc.getServletName();
    }
}

从源码中可以看出, GenericServlet 类将Servlet接口中其他的方法做了默认空实现,只将service()方法作为抽象,将来定义Servlet类时,可以继承GenericServlet,实现service()方法即可

  • HttpServlet是继承了GenericServlet的一个抽象类,源码片段如下

    package javax.servlet.http;
    
    import java.io.IOException;
    import java.io.PrintWriter;
    import java.io.OutputStreamWriter;
    import java.io.UnsupportedEncodingException;
    import java.lang.reflect.Method;
    import java.text.MessageFormat;
    import java.util.Enumeration;
    import java.util.Locale;
    import java.util.ResourceBundle;
    
    import javax.servlet.GenericServlet;
    import javax.servlet.ServletException;
    import javax.servlet.ServletOutputStream;
    import javax.servlet.ServletRequest;
    import javax.servlet.ServletResponse;
    
    public abstract class HttpServlet extends GenericServlet
        implements java.io.Serializable
    {
        private static final String METHOD_DELETE = "DELETE";
        private static final String METHOD_HEAD = "HEAD";
        private static final String METHOD_GET = "GET";
        private static final String METHOD_OPTIONS = "OPTIONS";
        private static final String METHOD_POST = "POST";
        private static final String METHOD_PUT = "PUT";
        private static final String METHOD_TRACE = "TRACE";
    
        private static final String HEADER_IFMODSINCE = "If-Modified-Since";
        private static final String HEADER_LASTMOD = "Last-Modified";
        
        private static final String LSTRING_FILE =
    	"javax.servlet.http.LocalStrings";
        private static ResourceBundle lStrings =
    	ResourceBundle.getBundle(LSTRING_FILE);
       
        
        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 / 1000 * 1000)) {
    		    // 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 {
    	    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);
    	}
      }
    }
    

    可以看到HttpServlet对GenericServlet类中的service方法进行了重写,后续我们只需要继承HttpServlet,重写其方法即可,这里我们用的最多的是doGet/doPost方法
    JavaWeb-Servlet笔记

6、ServletContext

servletContext,是Servlet中最大的一个接口,呈现了web应用的Servlet视图。

ServletContext实例是通过 getServletContext()方法获得的,由于HttpServlet继承GenericServlet的关系,GenericServlet类和HttpServlet类同时具有该方法。

web容器在启动的时候,它会为每个web程序都创建一个对应的ServletContext对象,它代表了当前的web应用

  • 共享数据

    import javax.servlet.ServletContext;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import java.io.IOException;
    
    public class ServletTest02 extends HttpServlet {
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            ServletContext servletContext = this.getServletContext();
    
            String username = "张三";
            servletContext.setAttribute("username",username);
        }
    
        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            doGet(req, resp);
        }
    }
    
    package com.wang.servlet;
    
    import javax.servlet.ServletContext;
    import javax.servlet.ServletException;
    import javax.servlet.http.HttpServlet;
    import javax.servlet.http.HttpServletRequest;
    import javax.servlet.http.HttpServletResponse;
    import java.io.IOException;
    
    public class GetServletContext extends HttpServlet {
    
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            ServletContext context = this.getServletContext();
            String username = (String) context.getAttribute("username");
    
            resp.setContentType("text/html");
            resp.setCharacterEncoding("utf-8");
            resp.getWriter().print("名字" + username);
        }
    
        @Override
        protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            doGet(req, resp);
        }
    }
    
        <servlet>
            <servlet-name>Servlet</servlet-name>
            <servlet-class>com.wang.servlet.ServletTest02</servlet-class>
        </servlet>
        <servlet-mapping>
            <servlet-name>Servlet</servlet-name>
            <url-pattern>/haha</url-pattern>
        </servlet-mapping>
    
       <servlet>
            <servlet-name>getServlet</servlet-name>
            <servlet-class>com.wang.servlet.GetServletContext</servlet-class>
        </servlet>
        <servlet-mapping>
            <servlet-name>getServlet</servlet-name>
            <url-pattern>/hello</url-pattern>
        </servlet-mapping>
    

JavaWeb-Servlet笔记
在ServletTest02中我们获取了一个servletContext对象,通过servletContext.setAttribute()方法传入了一个值,在GetServletContext中我们通过context.getAttribute()方法取出了这个值,servletContext可以保证共享数据,即我在一个Servlet中保存的数据,在另一个Servlet中可以拿到
JavaWeb-Servlet笔记

  • 获取初始化参数

      <context-param>
            <param-name>username</param-name>
            <param-value>zhangSan</param-value>
      </context-param>
    
     @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            ServletContext context = this.getServletContext();
            String username = context.getInitParameter("username");
            resp.getWriter().print(username);
        }
    
  • 请求转发

       public class GetServletContext extends HttpServlet {
    
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
            ServletContext context = this.getServletContext();
        }
    
    public class ServletTest02 extends HttpServlet {
        @Override
        protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
    
            System.out.println("进入了ServletTest02");
        }
    

    访问GetServletContext,控制台输出了,说明转发成功
    JavaWeb-Servlet笔记
    转发原理图
    JavaWeb-Servlet笔记

7、新增功能

  • Servlet 2.2:

    • 引入了 self-contained Web applications 的概念。
  • servlet 2.3:2000年10月份出来

    • Servlet API 2.3中最重大的改变是增加了 filters
    • Servlet 2.3 增加了 filters 和 filter chains 的功能。引入了 context 和 session listeners 的概念,当 context 或 session 被初始化或者被将要被释放的时候,和当向 context 或 session 中绑定属性或解除绑定的时候,可以对类进行监测。
  • servlet 2.4:2003年11月份推出

    • Servlet 2.4 加入了几个引起关注的特性,没有特别突出的新内容,而是花费了更多的功夫在推敲和阐明以前存在的一些特性上,对一些不严谨的地方进行了校验。

    • Servlet 2.4 增加了新的最低需求,新的监测 request 的方法,新的处理 response 的方法,新的国际化支持,RequestDispatcher 的几个处理,新的 request listener 类,session 的描述,和一个新的基于 Schema 的并拥有 J2EE 元素的发布描述符。这份文档规范全面而严格的进行了修订,除去了一些可能会影响到跨平台发布的模糊不清的因素。总而言之,这份规范增加了四个新类,七个新方法,一个新常量,不再推荐使用一个类。

    • 注意:改为 Schema 后主要加强了两项功能:

      • 元素不依照顺序设定;
      • 更强大的验证机制。
        • 主要体现:
          • 检查元素的值是否为合法的值
          • 检查元素的值是否为合法的文字字符或者数字字符
          • 检查 Servlet, Filter, EJB-ref 等等元素的名称是否单一的
    • 新增 Filter 四种设定:REQUEST、FORWARD、INCLUDE 和 ERROR。

    • 新增 Request Listener、Event和Request Attribute Listener、Event

    • 取消 SingleThreadModel 接口。当 Servlet 实现 SingleThreadModel 接口时,它能确保同时间内,只能有一个 thread 执行此 Servlet。

    • ServletRequest接口新增一些方法。

      • public String getLocalName();
      • public String getLocalAddr();
      • public int getLocalPort();
      • public int getRemotePort()
  • Servlet 2.5:2005 年 9 月发布 Servlet 2.5

    • 基于最新的 J2SE 5.0 开发的。
    • 支持 annotations 。
    • web.xml 中的几处配置更加方便。
    • 去除了少数的限制。
    • 优化了一些实例
  • Servlet 的各个版本对监听器的变化

    • Servlet 2.2 和 jsp1.1

      • 新增Listener:HttpSessionBindingListener
      • 新增Event: HttpSessionBindingEvent
    • Servlet 2.3 和 jsp1.2

      • 新增Listener:ServletContextListener,ServletContextAttributeListener,HttpSessionListener,HttpSessionActivationListener,HttpSessionAttributeListener
      • 新增Event: ServletContextEvent,ServletContextAttributeEvent,HttpSessionEvent
    • Servlet 2.4 和 jsp2.0

      • 新增Listener:ServletRequestListener,ServletRequestAttribureListener
      • 新增Event: ServletRequestEvent,ServletRequestAttributeEvent
    • Servlet 3.0

      Servlet 3.0 作为 Java EE 6 规范体系中一员,随着 Java EE 6 规范一起发布。该版本在前一版本(Servlet 2.5)的基础上提供了若干新特性用于简化 Web 应用的开发和部署。其中有几项特性的引入让开发者感到非常兴奋,同时也获得了 Java 社区的一片赞誉之声

      • 异步处理支持:有了该特性,Servlet 线程不再需要一直阻塞,直到业务处理完毕才能再输出响应,最后才结束该 Servlet 线程。在接收到请求之后,Servlet 线程可以将耗时的操作委派给另一个线程来完成,自己在不生成响应的情况下返回至容器。针对业务处理较耗时的情况,这将大大减少服务器资源的占用,并且提高并发处理速度。
      • 新增的注解支持:该版本新增了若干注解,用于简化 Servlet、过滤器(Filter)和监听器(Listener)的声明,这使得 web.xml 部署描述文件从该版本开始不再是必选的了。
      • 可插性支持:熟悉 Struts2 的开发者一定会对其通过插件的方式与包括 Spring 在内的各种常用框架的整合特性记忆犹新。将相应的插件封装成 JAR 包并放在类路径下,Struts2 运行时便能自动加载这些插件。Servlet 3.0 提供了类似的特性,开发者可以通过插件的方式很方便的扩充已有 Web 应用的功能,而不需要修改原有的应用。
    • Servlet 4.0

      从3.1到4.0将是对Servlet 协议的一次大改动,而改动的关键之处在于对HTTP/2的支持。HTTP2将是是继上世纪末HTTP1.1协议规范化以来首个HTTP协议新版本,相对于HTTP1.1,HTTP2将带来许多的增强。在草案提议中,Shing Wai列举出了一些HTTP2的新特性,而这些特性也正是他希望在Servlet 4.0 API中实现并暴露给用户的新功能,这些新特性如下: [4]

      • 请求/响应复用(Request/Response multiplexing)
      • 流的优先级(Stream Prioritization)
      • 服务器推送(Server Push)
      • HTTP1.1升级(Upgrade from HTTP 1.1)

8、编写线程安全的 Servlet 的建议:

  • 用方法的局部变量保存请求中的专有数据。对方法中定义的局部变量,进入方法的每个线程都有自己的一份方法变量拷贝。任何线程都不会修改其他线程的局部变量。如果要在不同的请求之间共享数据,应该使用会话来共享这类数据。
  • 只用 Servlet的成员变量来存放那些不会改变的数据。有些数据在 Servlet 生命周期中不发生任何变化,通常是在初始时确定的,这些数据可以使用成员变量保存,如数据库连接名称、其他资源的路径等。
  • 对可能被请求修改的成员变量同步。有时数据成员变量或者环境属性可能被请求修改。当访问这些数据时应该对它们同步,以避免多个线程同时修改这些数据。
  • 如果 Servlet 访问外部资源,那么需要同步访问这些资源。例如,假设 Servlet 要从文件中读写数据。当一个线程读写一个文件时,其他线程也可能正在读写这个文件。文件访问本身不是线程安全的,所以必须编写同步访问这些资源的代码。在编写线程安全的 Servlet 时,下面两种方法是不应该使用的:
    • 在 Servlet API 中提供了一个 SingleThreadModel 接口,实现这个接口的 Servlet 在被多个客户请求时一个时刻只有一个线程运行。这个接口已被标记不推荐使用。
    • 对 doGet() 或doPost() 方法同步。如果必须在 Servlet 中使用同步代码,应尽量在最小的代码块范围上进行同步。同步代码越小,Servlet 执行得才越好。

参考:Servlet百度百科

Servlet

上一篇:java Servlet注解实现过滤器


下一篇:maven 打包问题,war包生成目录不完整