前言

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)方法即可,如下代码:

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

  

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

@Override
public 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方法即可,如下代码:

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

  

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

@Override
public 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相关技术文章或行业资讯,欢迎大家关注和转发文章!

这些Servlet知识你一定要知道,金九银十大厂面试官都爱问的更多相关文章

  1. 两年经验拿到蚂蚁金服,字节offer,附上金九银十BAT面试核心知识点整理

    前言 我自己是本科毕业后在老东家干了两年多,老东家算是一家"小公司"(毕竟这年头没有 BAT 或 TMD 的 title 都不好意思报出身),毕业这两年多我也没有在大厂待过,因此找 ...

  2. 金九银十想去跳槽面试?那这份Java面经你真得看看了,写的非常详细!

    前言 前两天在和朋友吃饭的时候聊到时间这个东西是真的过的好坏啊,金三银四仿佛还在昨天.一眨眼金九银十又快到了,对程序员来说这两个是一年最合适的跳槽涨薪环节了,今年的你已经做好准备了吗?不妨看看这篇文章 ...

  3. 备战金九银十,Java研发面试题(Spring、MySQL、JVM、Mybatis、Redis、Tomcat)[带答案],刷起来!

    八月在即,马上就是"金九银十",又是跳槽招聘季.咱们这行公认涨薪不如跳槽加的快.但不建议频繁跳槽,还是要学会融合团队,抓住每个机会提升技能. 苏先生在这里给大家整理了一套各大互联网 ...

  4. 不等"金九银十",金风八月,我早已拿下字节跳动的offer

    字节跳动,我是在网上投的简历,之前也投过一次,简历都没通过删选,后来让师姐帮我改了一下简历,重新投另一个部门,获得了面试机会.7月23日,中午HR打电话过来预约了下午4点半面试,说会在线写代码,让我准 ...

  5. “金九银十”已过,总结我的天猫、蚂蚁、头条面试经历(Java岗)

    跳槽时时刻刻都在发生,但是我建议大家跳槽之前,先想清楚为什么要跳槽.切不可跟风,看到同事一个个都走了,自己也盲目的开始面试起来(期间也没有准备充分),到底是因为技术原因(影响自己的发展,偏移自己规划的 ...

  6. 金九银十跳槽高峰,面试必备之 Redis + MongoDB 常问80道面试题

    前言 有着“金九银十”之称的招聘旺季已经开启,跳槽高峰期也如约而至. 本文为主要是 Redis + MongoDB 知识点的攻略,希望能帮助到大家. 内容较多,大家准备好耐心和瓜子矿泉水. Redis ...

  7. 金九银十跳槽季,程序员面试点解析之Java专场

    前言 近年来Java工程师这个岗位炙手可热,市场需求大,学习Java的人也越来越多,所以IT企业与求职者的选择都比较多,那么IT企业在面试时都会提哪些问题呢.下面为大家分享 Java高级工程师面试阿里 ...

  8. Java程序员备战“金九银十”必备的面试技巧(附携程Java岗面试题)

    一.面试前的准备 1.1 如何准备一场面试1.1.1 如何获取大厂面试机会1.1.2 面试必知 ①. 准备介绍自己 ②. 关于着装 ③ .随身带上自己的成绩单和简历 ④. 如果笔试就提前刷一些笔试题 ...

  9. apk开发环境!多亏这份《秋招+金九银十-腾讯面试题合集》跳槽薪资翻倍!再不刷题就晚了!

    开头 最近很多网友反馈:自己从各处弄来的资料,过于杂乱.零散.碎片化,看得时候觉得挺有用的,但过个半天,啥都记不起来了.其实,这就是缺少系统化学习的后果. 为了提高大家的学习效率,帮大家能快速掌握An ...

随机推荐

  1. centos7 添加环境变量

    修改/etc/profile文件使其永久性生效,并对所有系统用户生效 PATH=$PATH:/usr/local/php/bin export PATH source /etc/profile

  2. origin生成直方图

    1. 导入数据 2. 选择一列,右键生成Frequent Count 3. 如果要显示相对频率,勾选Relative Frequency 4. 选择第一列和最后一列并生成柱状图 5. 双击生成的图形, ...

  3. 论文解读《ImageNet Classification with Deep Convolutional Neural Networks》

    这篇论文提出了AlexNet,奠定了深度学习在CV领域中的地位. 1. ReLu激活函数 2. Dropout 3. 数据增强 网络的架构如图所示 包含八个学习层:五个卷积神经网络和三个全连接网络,并 ...

  4. CentOS7 Nginx-1.10.3编译安装

    cat > nginx.sh <<EOF #停止apache,避免抢占端口号 systemctl stop httpd #创建nginx运行账户,非登录用户,不创建家目录 usera ...

  5. CodeForces 1408G Clusterization Counting

    题意 给定 \(n\) 个点的无向带权完全图,边权为 \(1\sim\frac{n(n-1)}{2}\).对于满足 \(1\leq k\leq n\) 的每个 \(k\) 求出将原图划分成 \(k\) ...

  6. MySQL全面瓦解5:数据操作-DML

    说明 DML(Data Manipulation Language)数据操作语言,是指对数据库进行增删改的操作指令,主要有INSERT.UPDATE.DELETE三种,代表插入.更新与删除,这是学习M ...

  7. kube-proxy实现原理

    1.service概念 service是一组pod的服务抽象,相当于一组pod的LB,负责将请求分发给对应的pod.service会为这个LB提供一个IP,一般称为cluster IP.kube-pr ...

  8. Charles使用part2——代理设置

    一.charles代理原理: 如果本地开了代理:  二.设置代理 1.设置代理端口: proxy->proxy setting 打开代理设置界面,代理端口默认是 8888,可以使用默认也可以自己 ...

  9. P4683 [IOI2008] Type Printer 打印机

    题意描述 [IOI2008] Type Printer 打印机 几百年前的 IOI 的题目还是很好的呀. 给你一个 诡异的 打印机,它只能用已有的字符来打印,而且必须每一个都用到.(这岂不是活字印刷术 ...

  10. 力扣 - 232. 用栈实现队列.md

    目录 题目 思路 代码实现 复杂度分析 题目 请你仅使用两个栈实现先入先出队列.队列应当支持一般队列的支持的所有操作(push.pop.peek.empty): 实现 MyQueue 类: void ...