前言

 Servlet是服务器端的Java应用程序,能够生产动静Web页面。透过JSP执行过程能够晓得JSP最终被编译成一个.class文件,查看该文件对应的Java类,发现该Java类继承自org.apache.jasper.runtime.HttpJspBase类,而HttpJspBase继承自HttpServlet类,由此可知JSP第一次运行时本质上是被JSP引擎翻译成了一个Servlet,而后再编译,最初再执行。

自定义Servlet类继 承HttpServlet抽象类,HttpServlet抽象类继承自GenericServlet抽象类,GenericServlet抽象类实现了Servlet、ServletConfig和Serializable接口

Servlet申明周期:

1、加载及实例化

Servlet容器负责加载和实例化Servlet。当客户端第一次(在web.xml文件中,通过load-on-startup标签能够配置Servlet,当web我的项目公布后立刻创立Servlet实例)  给服务器发送该Servlet申请时,Servlet容器会加载并创立Servlet实例,(留神:默认状况下不是Tomcat服务器或服务器上的Web利用启动的时候加载并实例化Servlet)。当客户端(能够是非第一次申请的客户端)再次向服务器发送该Servlet申请时,服务器会从内存中查找该Servlet实例,并用找到的Servlet实例解决用户申请。

在该过程中,Servlet容器会创立一个ServletConfig对象,该对象蕴含了Servlet的初始化配置信息。依据用户申请的URL地址,Servlet容器会依据配置信息查找该申请对应的Servlet类,由容器创立并治理该Servlet。

2、初始化

在Servlet容器实现Servlet类的实例化操作后,Servlet容器会调用Servlet的init()办法(在javax.servelt.Servlet接口中定义)对该Servlet进行初始化。对于每一个Servlet实例来说,init()办法只会被调用一次。初始化的目标是让Servlet在解决用户申请之前,做一些必要的筹备工作,例如建设数据库连贯,援用其余资源等。

3、解决申请

Servlet初始化之后,就处于就绪状态期待接管用户申请。当Servlet容器接管到客户端针对该Servlet的申请后,首先会针对这个申请创立ServletRequest和ServletResponse对象,之后调用Servlet的service()办法并把这两个参数传递给service()办法解决客户端申请。Servlet实例通过ServletRequest对象取得客户端的申请,通过调用ServletResponse对象的办法进行响应。申请处理完毕,ServletRequest和ServletResponse对象被销毁。

不论客户端发送申请的形式是Get还是POST,这个申请都由service办法来解决。在service办法的处理过程中,会依据客户端发送申请的形式不同,调用doGet和doPost办法别离进行解决,通过HttpServlet类中的service办法能够理解这一调用过程,如下代码:

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) {                doGet(req, resp);            } else {                long ifModifiedSince = req.getDateHeader(HEADER_IFMODSINCE);                if (ifModifiedSince < lastModified) {                    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);        }    }

4、销毁

销毁Servlet 由Servlet容器实现。默认状况下,用户第一次发送Servlet申请,该Servlet加载、实例化、初始化、解决用户申请,当申请处理完毕后,该Servlet通常状况下驻留在内存中,期待解决下一个针对该Servlet的申请。当下一个针对该Servlet的申请达到时,间接从内存中获取该Servlet实例并对该申请进行解决。如果Tomcat这个Web应用服务器敞开(服务器上所有的Web利用都敞开),或者该Servlet所在的Web利用敞开,该Servlet实例会被销毁。

Web利用被敞开时,Servlet容器会先调用Servlet实例的destroy办法,而后再销毁Servlet实例,同时也会销毁与Servlet相关联的ServletConfig对象。程序员通常在destroy()办法的实现中开释该Servlet所占用的资源,如敞开数据库连贯,敞开文件输出/输入流等。

通过Servlet申明周期能够晓得所创立的Servlet对象属于单例。

Servlet2.X配置

在web.xml文件中,通过在<web-app>节点下配置servlet元素和servlet-mapping元素,把用户拜访的URL映射到指定的Servlet类,如下代码:

<web-app>  <!-- 省略其余配置 -->  <servlet>      <!-- servlet-name指定Servlet名,要与上面servlet-mapping元素中的servlet-name保持一致 -->    <servlet-name>doLogin</servlet-name>    <!-- servlet-class对应着Servlet类齐全限定名 -->    <servlet-class>com.jd.serlvet.LoginServlet</servlet-class>  </servlet>    <servlet-mapping>      <!-- servlet-name要与下面servlet元素中的servlet-name保持一致 -->    <servlet-name>doLogin</servlet-name>    <!-- url-pattern设定以后Servlet在浏览器中运行时的url -->    <url-pattern>/doLogin</url-pattern>  </servlet-mapping></web-app>

下面采纳了准确匹配的模式配置了URL到Servlet之间的映射关系,接下来介绍两种非准确匹配的Servlet配置形式:

<!-- 对doLogin门路下的所有申请都由doLogin对应的Servlet类进行解决 -->  <servlet-mapping>    <servlet-name>doLogin</servlet-name>    <url-pattern>/doLogin/*</url-pattern>  </servlet-mapping>    <!-- 对所有以.do为后缀的申请都由doLogin对应的Servlet类进行解决 -->    <servlet-mapping>    <servlet-name>doLogin</servlet-name>    <!-- 不能为/*.do -->    <url-pattern>*.do</url-pattern>  </servlet-mapping>

在配置了URL与Servlet的映射后,当Servlet容器收到一个申请时,首先确定是由哪个Web利用响应这个申请,而后从该Web利用的web.xml文件中查找URL对应的Servlet类进行解决。

Servlet初始化参数设置

在web.xml文件中配置Servlet时,还能够在servlet元素中增加init-param元素事后对Servlet进行初始化设置,当Servlet加载时即可从该Servlet配置文件中获取初始化参数。

  <!-- 省略其余配置 -->  <servlet>    <servlet-name>doLogin</servlet-name>    <servlet-class>com.jd.serlvet.LoginServlet</servlet-class>    <!-- 配置多个初始化参数,则须要写多个init-param元素 -->    <init-param>            <param-name>name</param-name>            <param-value>Tom</param-value>    </init-param>  </servlet>    <servlet-mapping>    <servlet-name>doLogin</servlet-name>    <url-pattern>/doLogin</url-pattern>  </servlet-mapping>

如何获取:

a、在无参init办法中间接调用getInitParameter(String name)办法即可,如下代码:

@Overridepublic void init() throws ServletException {    String name = getInitParameter("name");    System.out.println(name);}

b、在参数为ServletConfig的办法中调用ServletConfig内getInitParameter(String name)办法即可,如下代码:

@Overridepublic void init(ServletConfig config) throws ServletException {    String name = config.getInitParameter("name");    System.out.println(name);}

Servlet上下文(环境对象)初始化参数设置

有时候不仅须要针对单个Servlet进行初始化参数设置,还须要对蕴含该Web援用中所有Servlet的环境对象进行初始化参数设置,使该参数能被所有的Servlet共享,如下代码:

  <!-- 省略其余配置 -->  <!-- 配置多个初始化参数,则须要写多个context-param元素 -->   <context-param>       <param-name>name</param-name>       <param-value>Tom</param-value>   </context-param>   <servlet>    <servlet-name>doLogin</servlet-name>    <servlet-class>com.jd.serlvet.LoginServlet</servlet-class>    <load-on-startup>1</load-on-startup>  </servlet>    <servlet-mapping>    <servlet-name>doLogin</servlet-name>    <url-pattern>/doLogin</url-pattern>  </servlet-mapping></web-app>

如何获取:

a、在无参init办法中间接调用getServletContext ()办法获取Servlet上下文对象,而后应用该对象调用getInitParameter办法即可,如下代码:

@Overridepublic void init() throws ServletException {    String name = getServletContext ().getInitParameter("name");    System.out.println(name);}

b、在参数为ServletConfig的办法中调用ServletConfig内getServletContext ()办法获取Servlet上下文对象,而后应用该对象调用getInitParameter办法即可,如下代码:

@Overridepublic void init(ServletConfig config) throws ServletException {    ServletContext servletContext = config.getServletContext();    String name = servletContext.getInitParameter("name");    System.out.println(name);}

Servlet 3.0

Servlet API蕴含javax.servlet和javax.servlet.http两个包,从Servlet 3.0开始,为了实现Servlet3.0的一些新个性,又减少了javax.sevlet.annotation和javax.sevlet.descriptor两个包,Tomcat服务器必须是7.0及其以上版本

Servlet3.0的重大变革之一是反对注解,通过应用注解定义并部署Servlet,程序员毋庸在web.xml文件中配置Servlet,如下代码:

@WebServlet(name="doLogin",urlPatterns="/doLogin",initParams={@WebInitParam(name="name",value="Tom")})public class LoginServlet extends HttpServlet {    private static final long serialVersionUID = 1L;        protected void doPost(HttpServletRequest request, HttpServletResponse response)            throws ServletException, IOException {        response.sendRedirect("success.jsp");    }}

 阐明:

name属性:指定Servlet名,相似于web.xml中servlet-name元素;

urlPatterns属性:指定拜访URL,相似于web.xml中的url-pattern元素;能够是数组,URL之间应用逗号距离。

initParams属性:设置初始化参数,该属性中应用@WebInitParam注解设置单个初始化参数;在@WebInitParam注解中,name属性指定参数名,相似于web.xml中的param-name元素;value属性指定参数值,相似于web.xml中的param-value元素。能够在initParams属性中配置多个@WebInitParam注解,用于初始化多个参数。 

wb.xml:

<!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN" "http://java.sun.com/dtd/web-app_2_3.dtd" > <web-app>  <display-name>Archetype Created Web Application</display-name>  <context-param>          <param-name>name</param-name>          <param-value>lucy</param-value>  </context-param>  <servlet>      <servlet-name>TestServlet</servlet-name>      <display-name>TestServlet</display-name>      <description></description>      <servlet-class>com.jd.servlet.TestServlet</servlet-class>      <init-param>          <param-name>mobile</param-name>          <param-value>120</param-value>      </init-param>      <load-on-startup>1</load-on-startup>  </servlet>  <servlet-mapping>      <servlet-name>TestServlet</servlet-name>      <url-pattern>/TestServlet</url-pattern>  </servlet-mapping></web-app>

TestServlet.java:

package com.jd.servlet; import java.io.IOException; import javax.servlet.ServletConfig;import javax.servlet.ServletException;import javax.servlet.ServletRequest;import javax.servlet.ServletResponse;import javax.servlet.http.HttpServlet;import javax.servlet.http.HttpServletRequest;import javax.servlet.http.HttpServletResponse; /** * Servlet implementation class TestServlet */public class TestServlet extends HttpServlet {    private static final long serialVersionUID = 1L;       //Servlet对象属于单实例(程序运行完结之前,缓存中只有一个对象存在)    public TestServlet() {        //用于为成员变量赋值,会触发对象创立:默认状况下第一次应用该Servlet时创建对象执行;        //web.xml中退出<load-on-startup>1</load-on-startup>,在tomcat启动时对象创立,TestServlet执行,随之init()办法也立刻执行                super();        System.out.println("TestServlet"+this);           }     //    @Override    public void destroy() {        super.destroy();         System.out.println("destroy"+this);    }      @Override    public void init() throws ServletException {//初始化;对于每一个Servlet实例来说,创建对象执行init()办法,并且只会被调用一次        super.init();        String name=getServletContext().getInitParameter("name");//调用公共参数        System.out.println("1111111"+name);        System.out.println("init())"+this);    }         @Override    public void init(ServletConfig config) throws ServletException {        super.init(config);        String name=config.getServletContext().getInitParameter("name");//调用公共参数        System.out.println("你好"+name);        String mobile=config.getInitParameter("mobile");//调用私人参数        System.out.println("快打"+mobile);        System.out.println("init(config))"+this);    }             //申请最先达到service判断执行哪个办法,doGet还是doPost    @Override    protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {        super.service(req, resp);        System.out.println("service"+this);    }     //a标签,form表单method是 get办法   异步默认type是get办法    protected void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {         System.out.println("doGet"+this);    }    //form表单method是 post办法  异步指定type是post办法    protected void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException {         System.out.println("doPost"+this);    } }

最初

感激你看到这里,看完有什么的不懂的能够在评论区问我,感觉文章对你有帮忙的话记得给我点个赞,每天都会分享java相干技术文章或行业资讯,欢送大家关注和转发文章!