JavaWeb开发基础Servlet生命周期与工作原理

article/2025/8/6 3:34:57

Servlet生命周期

Servlet的生命周期由Servlet容器(如Tomcat、Jetty等)管理,主要包括以下5个阶段:

  1. 加载Servlet类

  2. 创建Servlet实例

  3. 调用init方法

  4. 调用service方法

  5. 调用destroy方法

  1. 加载(Loading):

    • 当Servlet容器启动或第一次接收到对某个Servlet的请求时,它会加载Servlet类
    • 这个过程只发生一次,除非Servlet被卸载或容器重新启动
  2. 实例化(Instantiation):

    • 在加载Servlet类之后,容器会创建一个Servlet实例
    • 这个实例化过程也是一次性的,除非Servlet被卸载或容器重新启动
  3. 初始化(Initialization):

    • 容器调用Servlet实例的init()方法进行初始化

    • init()方法只被调用一次,用于执行一次性的初始化任务,如读取配置参数、初始化资源等

    • 你可以重写init()方法以执行自定义的初始化逻辑

      public void init(ServletConfig config)
  4. 请求处理(Request Handling):

    • 每当有请求到达时,容器会调用Servlet的service()方法来处理请求

    • service()方法根据请求的类型(如GET、POST等)调用相应的处理方法(如doGet()doPost()等)

      public void service(ServletRequest request, ServletResponse response)
  5. 销毁(Destruction):

    • 当容器决定卸载Servlet(如容器关闭或Servlet被重新加载)时,它会调用Servlet的destroy()方法

    • destroy()方法用于执行清理任务,如关闭资源、保存状态等

    • destroy()方法只被调用一次

      public void destroy()

Servlet重新加载

Servlet被重新加载通常发生在以下几种情况下:

  1. Servlet类文件被修改:

    • 如果在开发过程中修改了Servlet类文件,Servlet容器可能会检测到这些更改并重新加载Servlet
    • 这通常发生在开发环境中,生产环境中一般不会自动重新加载
  2. 容器配置发生变化:

    • 如果修改了Servlet容器的配置文件(如web.xml),容器可能会重新加载相关的Servlet
  3. 容器重启:

    • 当Servlet容器重启时,所有的Servlet都会被重新加载
  4. 手动重新加载:

    • 某些Servlet容器提供了手动重新加载Servlet的功能,通常通过管理控制台或命令行工具

当Servlet被重新加载时,Servlet容器会执行以下步骤:

  1. 销毁现有的Servlet实例:

    • 容器调用现有Servlet实例的destroy()方法,执行清理任务
    • 释放资源,如关闭数据库连接、释放文件句柄等
  2. 卸载Servlet类:

    • 容器卸载现有的Servlet类,释放类加载器持有的资源
  3. 重新加载Servlet类:

    • 容器重新加载Servlet类文件,创建新的类加载器实例
  4. 实例化新的Servlet对象:

    • 容器创建新的Servlet实例
  5. 初始化新的Servlet实例:

    • 容器调用新的Servlet实例的init()方法进行初始化

Servlet代码示例

有3种方式使用Servlet:

①实现Servlet接口

import java.io.*;
import javax.servlet.*;public class First implements Servlet {ServletConfig config = null;public void init(ServletConfig config) {this.config = config;System.out.println("servlet is initialized");}public void service(ServletRequest req, ServletResponse res)throws IOException, ServletException {res.setContentType("text/html");PrintWriter out = res.getWriter();out.print("<html><body>");out.print("<b>hello simple servlet</b>");out.print("</body></html>");}public void destroy() {System.out.println("servlet is destroyed");}public ServletConfig getServletConfig() {return config;}public String getServletInfo() {return "copyright 2007-1010";}
}

②继承GenericServlet类

import java.io.*;
import javax.servlet.*;public class First extends GenericServlet {public void service(ServletRequest req, ServletResponse res)throws IOException, ServletException {res.setContentType("text/html");PrintWriter out = res.getWriter();out.print("<html><body>");out.print("<b>hello generic servlet</b>");out.print("</body></html>");}
}

③继承HttpServlet类

import javax.servlet.http.*;
import javax.servlet.*;
import java.io.*;public class DemoServ extends HttpServlet {public void doGet(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {res.setContentType("text/html");PrintWriter pw = res.getWriter();String name = req.getParameter("name");pw.println("Welcome " + name);}
}

继承HttpServlet类是最常用的方式,接下来通过六个步骤,看看具体实践。

第一步,创建目录结构。

第二步,创建Servlet。

import javax.servlet.http.*;
import javax.servlet.*;
import java.io.*;public class DemoServlet extends HttpServlet {public void doGet(HttpServletRequest req, HttpServletResponse res)throws ServletException, IOException {res.setContentType("text/html");PrintWriter pw = res.getWriter(); pw.println("<html><body>");pw.println("Welcome to servlet");pw.println("</body></html>");pw.close();}
}

第三步,编译Servlet。

<dependency><groupId>javax.servlet</groupId><artifactId>javax.servlet-api</artifactId><version>3.1.0</version><scope>provided</scope>
</dependency>

编译后将class文件拷贝到WEB-INF/classes。

第四步,创建web.xml。

<web-app><servlet><servlet-name>sonoojaiswal</servlet-name><servlet-class>DemoServlet</servlet-class></servlet><servlet-mapping><servlet-name>sonoojaiswal</servlet-name><url-pattern>/welcome</url-pattern></servlet-mapping>
</web-app>

web.xml是一个部署描述符文件,用于配置Java Web应用程序。

Java Servlet 3.0引入了@WebServlet注解,用于简化Servlet的配置:

import javax.servlet.http.*;
import javax.servlet.*;
import java.io.*;
import javax.servlet.annotation.WebServlet;@WebServlet("/welcome")
public class DemoServlet extends HttpServlet {private static final long serialVersionUID = 1L;public void doGet(HttpServletRequest req, HttpServletResponse res) throws ServletException, IOException {res.setContentType("text/html");PrintWriter pw = res.getWriter(); pw.println("<html><body>");pw.println("Welcome to servlet");pw.println("</body></html>");pw.close();}
}

web.xml在低版本Servlet是必须的,在高版本Servlet已经不再需要。

第五步,部署Servlet。

将项目文件复制到Apache Tomcat的webapps目录下,启动Tomcat服务。

第六步,访问Servlet。

浏览器请求http://localhost:9999/demo/welcome,就能看到Servlet响应内容。

Servlet工作原理

  1. 请求映射:

    • 浏览器发送一个HTTP请求到服务器

    • 服务器根据web.xml文件中的配置,将请求映射到相应的Servlet

  2. 创建请求和响应对象:

    • 服务器为每个请求创建HttpServletRequestHttpServletResponse对象。这些对象包含了请求的所有信息和用于生成响应的方法
  3. 调用service方法:

    • 服务器在一个新的线程中调用Servlet的service方法来处理请求
  4. 调用public service方法:

    • public service 方法内部会调用protected service方法。public service方法是HttpServlet类的一部分,它负责处理所有类型的HTTP请求

      public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {if (req instanceof HttpServletRequest && res instanceof HttpServletResponse) {HttpServletRequest request = (HttpServletRequest)req;HttpServletResponse response = (HttpServletResponse)res;this.service(request, response);} else {throw new ServletException("non-HTTP request or response");}
      }
  5. 调用protected service方法:

    • protected service方法根据请求的类型(如GET、POST等)调用相应的处理方法(如doGetdoPost等)

      protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {String method = req.getMethod();long lastModified;if (method.equals("GET")) {lastModified = this.getLastModified(req);if (lastModified == -1L) {this.doGet(req, resp);} else {long ifModifiedSince = req.getDateHeader("If-Modified-Since");if (ifModifiedSince < lastModified) {this.maybeSetLastModified(resp, lastModified);this.doGet(req, resp);} else {resp.setStatus(304);}}} else if (method.equals("HEAD")) {lastModified = this.getLastModified(req);this.maybeSetLastModified(resp, lastModified);this.doHead(req, resp);} else if (method.equals("POST")) {this.doPost(req, resp);} else if (method.equals("PUT")) {this.doPut(req, resp);} else if (method.equals("DELETE")) {this.doDelete(req, resp);} else if (method.equals("OPTIONS")) {this.doOptions(req, resp);} else if (method.equals("TRACE")) {this.doTrace(req, resp);} else {String errMsg = lStrings.getString("http.method_not_implemented");Object[] errArgs = new Object[]{method};errMsg = MessageFormat.format(errMsg, errArgs);resp.sendError(501, errMsg);}
      }
  6. 调用doGet方法:

    • 如果请求类型是GET,protected service方法会调用doGet方法。doGet方法处理请求并生成响应
  7. 生成响应并发送给客户端:

    • doGet方法生成响应(如HTML、JSON等),并通过HttpServletResponse对象将响应发送回客户端
  8. 删除请求和响应对象:

    • 在响应发送完毕后,Web容器会删除HttpServletRequestHttpServletResponse对象,以释放资源
    • 处理请求的线程要么被返回到线程池,要么被删除,这取决于服务器的实现

总结,Servlet的生命周期包括初始化(init)、请求处理(service,调用doGet、doPost等)、和销毁(destroy)。工作原理是:客户端请求,服务器分发,创建请求和响应对象,调用service方法,生成并发送响应,删除请求和响应对象。

参考资料:

https://www.javatpoint.com/life-cycle-of-a-servlet

https://www.javatpoint.com/steps-to-create-a-servlet-using-tomcat-server

https://www.javatpoint.com/how-servlet-works

ChatGPT


http://www.hkcw.cn/article/DcYodYGcrY.shtml

相关文章

Electron-vite【实战】MD 编辑器 -- 系统菜单(含菜单封装,新建文件,打开文件,打开文件夹,保存文件,退出系统)

最终效果 整体架构 src/main/index.ts import { createMenu } from ./menu在 const mainWindow 后 // 加载菜单createMenu(mainWindow)src/main/menu.ts import { BrowserWindow, Menu, MenuItem, MenuItemConstructorOptions, dialog, shell } from electron import fs from…

天气预报中的AI:更准确的预测如何实现

如今的天气预报早已不是简单的看云识天气&#xff0c;而是变成了一场数据与算法的科技博弈。当你在手机App上查看未来两小时的降雨概率时&#xff0c;背后可能是AI模型分析了全球数万颗气象卫星的数据&#xff1b;当你收到台风路径预警短信时&#xff0c;或许是AI提前五天就锁定…

虚拟化数据恢复—XenServer虚拟机虚拟磁盘文件丢失的数据恢复案例

虚拟化环境&#xff1a; 某品牌720服务器中有一组通过型号为H710P的RAID卡4块STAT硬盘组建的RAID10&#xff0c;上层部署Xen Server服务器虚拟化平台。虚拟机安装的Windows Server系统&#xff0c;运行Web服务器。有系统盘 数据盘两个虚拟机磁盘。 虚拟化故障&#xff1a; 机…

Java 之殇:从中流砥柱到“被温柔替代”

—— 一位老派 Java 工程师的自述 今天看到一篇江苏的作者发出的《公司Rust团队全员被裁&#xff0c;只因把服务写得「太稳定」&#xff1a;“项目0故障、0报警&#xff0c;那养着3个Rust工程师没用啊”》帖子。看到那篇文章第一反应也是&#xff1a;这八成是 AI 编的。但说实…

vscode一直连接不上虚拟机或者虚拟机容器怎么办?

1. 检查并修复文件权限 右键点击 C:\Users\20325\.ssh\config 文件&#xff0c;选择 属性 → 安全 选项卡。 确保只有你的用户账户有完全控制权限&#xff0c;移除其他用户&#xff08;如 Hena\Administrator&#xff09;的权限。 如果 .ssh 文件夹权限也有问题&#xff0c;同…

面试中的项目经验考查:如何让实战经历成为你的决胜王牌

阅读原文 "你在项目中遇到的最大困难是什么&#xff1f;" 当面试官抛出这个问题时&#xff0c;你是否曾感到一阵心虚&#xff1f;是否担心自己的回答显得单薄无力&#xff1f;在竞争激烈的技术岗位面试中&#xff0c;项目经验往往是决定成败的关键因素。资深HR甚至建…

基于Java(SSH框架)+MySQL 实现(Web)公司通用门户(CMS)网站

一、公司通用门户网站的设计与实现 摘要&#xff1a;随着IT应用的深入普及&#xff0c;各行各业都积累了大量的信息资源&#xff0c;实现企业内部信息技术资源的有效整合和精益化管理&#xff0c;是越来越多公司企业的迫切需求。公司门户网站是一个企业向外宣传企业品牌和展示…

vue3实现鼠标悬浮div动画效果

需求 鼠标悬浮在div上显示下载按钮和信息&#xff0c;同时保持下面的div位置不变&#xff1b;当鼠标移走的时候就隐藏恢复原样。 效果&#xff1a; 代码 <script setup> const software ref([{id: "one",title: "软件",container: [{id: "123…

数据结构与算法之单链表面试题(新浪、百度、腾讯)

单链表面试题&#xff08;新浪、百度、腾讯&#xff09; 求单链表中的有效节点的个数 public int getCount(HeroNode head) {Hero1 cur head.getNext();int count 0;while(cur ! null) {count;cur cur.getNext();}return count;}查找单链表中的倒数第k个结点【新浪面试题】…

Google Play推出新功能:用户可直接向Gemini提问应用相关问题

5 月 30 日消息&#xff0c;谷歌在Google Play中广泛推出了由 Gemini AI 提供支持的“向Google Play询问此应用”功能&#xff0c;该功能已正式出现在Google Play的46.1.39-31 版本中。 “向Google Play询问此应用”这项功能&#xff0c;将 Gemini AI 直接集成到Google Play中&…

PyTorch学习(1):张量(Tensor)核心操作详解

PyTorch学习(1)&#xff1a;张量&#xff08;Tensor&#xff09;核心操作详解 一、张量&#xff08;Tensor&#xff09;核心操作详解 张量是PyTorch的基础数据结构&#xff0c;类似于NumPy的ndarray&#xff0c;但支持GPU加速和自动微分。 1. 张量创建与基础属性 import to…

农村土地承包经营权二轮延包—生成地块的KJZB字段

"关于地块的空间坐标&#xff08;KJZB&#xff09;字段&#xff0c;可能稍微复杂一点&#xff0c;用脚本生成较好。空间坐标&#xff0c;目前有两种表达&#xff1a;方案一&#xff0c;根据地块上界址点的个数依次填上&#xff08;如4个为J1/J2/J3/J4&#xff09;&#xf…

时空数据智能分析的原理和案例分享

在当今数字化时代,时空数据如同隐藏在海量信息中的宝藏,蕴含着丰富的价值,等待我们去挖掘和利用。从城市交通的实时监测与优化,到自然灾害的预警与防范,从精准农业的智能管理,到金融市场的动态分析,时空数据的身影无处不在,深刻地影响着我们生活的方方面面。DeepSeek,…

专场回顾 | 重新定义交互,智能硬件的未来设计

自2022年起&#xff0c;中国智能硬件行业呈现出蓬勃发展的态势&#xff0c;市场规模不断扩大。一个多月前&#xff0c;“小智AI”在短视频平台的爆火将智能硬件带向了大众视野&#xff0c;也意味着智能硬件已不再仅仅停留在概念和技术层面&#xff0c;而是加速迈向实际落地应用…

解决访问网站提示“405 很抱歉,由于您访问的URL有可能对网站造成安全威胁,您的访问被阻断”问题

一、问题描述 本来前几天都可以正常访问的网站&#xff0c;但是今天当我们访问网站的时候会显示“405 很抱歉&#xff0c;由于您访问的URL有可能对网站造成安全威胁&#xff0c;您的访问被阻断。您的请求ID是&#xff1a;XXXX”&#xff0c;而不能正常的访问网站&#xff0c;如…

十二、【核心功能篇】测试用例列表与搜索:高效展示和查找海量用例

【核心功能篇】测试用例列表与搜索&#xff1a;高效展示和查找海量用例 前言准备工作第一步&#xff1a;更新 API 服务以支持分页和更完善的搜索第二步&#xff1a;创建测试用例列表页面组件 (src/views/testcase/TestCaseListView.vue)第三步&#xff1a;测试列表、搜索、筛选…

Windows环境下PHP,在PowerShell控制台输出中文乱码

解决方法&#xff1a; 以管理员运行PowerShell , 输入&#xff1a; chcp 65001 重启控制台&#xff1b;然后就正常输出中文&#xff1b;

安卓apk安装包签名步骤

1.获取apk对应的原始证书&#xff08;问前端要&#xff09; 2.打开命令窗口win r 输入 cmd 3.输入 cd .android 定位到 .android 文件夹 4.执行证书签名命令 keytool -genkey -v -keystore 前端提供的.keystore -alias 自定义别名信息 -keyalg RSA -validity 10000 密钥为&a…

C与C++相互调用

C与C为什么相互调用的方式不同 C 和 C 之间的相互调用方式存在区别&#xff0c;主要是由于 C 和 C 语言本身的设计和特性不同。 函数调用和参数传递方式不同 &#xff1a; C 和 C 在函数调用和参数传递方面有一些不同之处。 C 使用标准 的函数调用约定&#xff0c;而 …

Nest全栈到失业(附加):Mysql+TypeOrm构建CRUD

前置内容 在此之前,我希望你准备好一个docker环境,以及魔法的网络哦 自己创建一个项目哈,使用nest new XXX Docker 什么是docker?相信很多人都知道了,说白了,就是一个镜像容器;以mysql为例,你在电脑上使用mysql5.6啥的,他电脑上是5.7啥的,然后数据内容不兼容了,怎么办了?他卸…