Tomcat 学习之 Servlet

news2025/5/22 18:32:28

目录

1 Servlet 介绍

2 创建一个 Servlet

3 web.xml 介绍(不涉及 filter 和 listener 标签)

3.1 display-name

3.2 welcome-file-list

3.3 servlet

3.4 session-config

3.5 error-page

4 Tomcat 如何根据 URL 定位到 Servlet

5 执行 Servlet

5.1 Servlet 接口(规范)

5.2 Servlet 生命周期

5.3 执行 init 初始化方法

5.4 执行 service 方法


最近学习 Servlet 的时候,不是那么认真,认为它已经过时了,可以不用学。直到学习 SpringMVC 看到 DispatcherServlet 的时候,才发现自己错了,JavaWeb 三大组件 (Servlet、Filter、Listener)值得学习。

1 Servlet 介绍

        互联网兴起之初,当时的 Sun 公司(后来被 Oracle 收购)已然看到了这次机遇,为了使 Java 进军 Web 编程领域,推出了 Applet 支持 Web 应用,但是效果并不如预想的那般美好,Applet 并没有给业界带来多大影响。之后,Sun 公司的大佬潜心专研,推出了 Servlet(Server Applet,全称 Java Servlet)实现动态网页。Tomcat 的前身为 Catalina,Catalina 是一个轻量级的 Servlet 容器,提供了运行 Servlet 的环境和服务。

       

上图是一个粗略的 Servlet 执行过程表示图,从 Servlet 容器(Tomcat)启动到运行 Servlet 的过程非常复杂,本文只介绍 Servlet 是如何使用的。那么 Servlet 是用来干什么的呢?

Servlet 主要用来处理客户端的请求并生成响应,分为以下几点:(来自 GPT)

  1. 接收和处理客户端的 HTTP 请求:Servlet 可以接收来自客户端的各种类型的 HTTP 请求,例如 GET、POST、PUT、DELETE 等,并根据请求的类型和参数生成相应的响应。

  2. 生成动态内容:Servlet 可以根据请求的参数和服务器端的数据生成动态的 HTML 页面、XML 文档、JSON 数据等响应内容,从而实现动态网页的效果。

  3. 处理表单数据:Servlet 可以接收和处理客户端提交的表单数据,例如用户注册、登录、查询等操作。

  4. 实现会话管理:Servlet 可以通过 HTTP 会话(Session)来跟踪和管理客户端与服务器之间的会话状态,例如存储用户登录信息、购物车信息等。

  5. 实现安全控制:Servlet 可以通过验证用户身份、授权访问权限等方式实现安全控制,保护服务器和客户端的数据安全。

  6. 与其他 Web 组件交互:Servlet 可以与其他 Web 组件(如 JSP、Filter、Listener 等)进行交互,实现复杂的 Web 应用程序。

2 创建一个 Servlet

创建一个 JavaWeb 工程

        创建过程请参考:IDEA 2023.2 配置 JavaWeb 工程-CSDN博客

导入 Servlet-api.jar 包

        Maven 中通过配置 pom.xml 导入

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

        非 Maven 导入 Servlet-api.jar

点击 File,再点击 Project Structrue

选择 Libraries,点击 + 号,选择 Java

找到 Tomcat 中 lib 目录下的 servlet-api.jar,点击 OK

先点击 Apply,再点击 OK

工程目录

在 src 目录下创建一个 com.test 包,在其中创建 TestServlet 类

package com.test;


import javax.servlet.*;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import java.io.IOException;

public class TestServlet implements Servlet {

    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        // 接口 HttpServletRequest 实现 servletRequest 接口
        HttpServletRequest httpServletRequest = (HttpServletRequest) servletRequest;
        String method = httpServletRequest.getMethod();
        // 获得请求方法
        if (method.equals("GET")) {
            this.doGet();
        } else if (method.equals("POST")) {
            this.doPost();
        }
    }

    public void doGet(){
        System.out.println("GET 请求");
    }

    public void doPost(){
        System.out.println("POST 请求");
    }

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

    @Override
    public void destroy() {

    }

    @Override
    public void init(ServletConfig servletConfig) throws ServletException {

    }

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

修改 web.xml 文件

<?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">

    <!-- 配置 Servlet 程序 -->
    <servlet>
        <!-- 设置 Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <!-- 设置 Servlet 程序所在路径(位置) -->
        <servlet-class>com.test.TestServlet</servlet-class>
    </servlet>

    <!-- 设置 Servlet 映射-->
    <servlet-mapping>
        <!-- Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <url-pattern>/test</url-pattern>
    </servlet-mapping>
</web-app>

启动 Tomcat

哈哈,404 ,这时因为没有设置“首页文件”

输入 localhost:8080/servlettest/test 啥也没有

在 IDEA 中可以看到服务器已经收到了请求并响应了

在浏览器按 F12、刷新网页、查看网络,可以看到请求的消息,至此 Servlet 学习结束了

3 web.xml 介绍(不涉及 filter 和 listener 标签)

3.1 display-name

<display-name> 元素常用于配置 servlet、过滤器或其他 Web 组件的显示名称。这个显示名称主要用于在管理界面或日志中标识该组件,以方便识别和管理

     <!-- <display-name> 元素常用于配置 servlet、过滤器或其他 Web 组件的显示名称。
    这个显示名称主要用于在管理界面或日志中标识该组件,以方便识别和管理。 -->
    <!-- 标识项目名 -->
    <display-name>ServletTest</display-name>

3.2 welcome-file-list

welcome-file-list 用来定义首页文件,也就是用户直接输入域名时跳转的页面(如http://localhost:8080/)

        在 tomact 的 conf 目录下,web.xml 使用了该标签,所以我们在 web 目录下创建的 index.jsp 文件在 Tomcat 服务器启动时会被执行。此外,成功执行了第一个首页文件,后面的首页文件不会执行,如果没有定义首页文件,则显示 404

    <!-- 定义首页文件,也就是用户直接输入域名时跳转的页面(如http://localhost:8080/)-->
    <welcome-file-list>
        <welcome-file>index.jsp</welcome-file>
        <welcome-file>test.jsp</welcome-file>
    </welcome-file-list>

3.3 servlet

    <!-- 配置 Servlet 程序 -->
    <servlet>
        <!-- 放在第一句不会报错 -->
        <display-name>TestServlet</display-name>
        <!-- 设置 Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <!-- 设置 Servlet 程序所在路径(位置) -->
        <servlet-class>com.test.TestServlet</servlet-class>
        <!-- 设置 Servlet 初始化参数
        可以通过 ServletConfig.getInitParamenter(String name) 方法访问初始化参数 -->
        <init-param>
            <param-name>driver</param-name>
            <param-value>com.mysql.jdbc.Driver</param-value>
        </init-param>
        <init-param>
            <param-name>url</param-name>
            <param-value>jdbc:mysql://localhost:3306/study</param-value>
        </init-param>
        <init-param>
            <param-name>username</param-name>
            <param-value>root</param-value>
        </init-param>
        <init-param>
            <param-name>passwd</param-name>
            <param-value>123456</param-value>
        </init-param>
        <!-- load-on-startup 指定一个 servlet 的加载顺序
         当值为 0 或者大于 0 时,表示容器在应用启动时就加载这个 servlet
         当值是一个负数时或者没有指定时,表示容器在该 servlet 被选择时才加载
         正数的值越小,启动该servlet的优先级越高-->
        <load-on-startup>1</load-on-startup>
        <!--支持异步处理-->
        <async-supported>true</async-supported>
    </servlet>

其中我觉得最重要的点是 url-pattern

参考文档:tomcat关于配置servlet的url-pattern的问题详解 - sogeisetsu - 博客园 (cnblogs.com)

精确匹配(具体的某个文件)

如通过 http://localhost:8080/servlettest/img/2.jpg 访问工程 servlettest 中的 img 目录中的 2.jpg 图片,会使用 TestServlet 处理,不会正常访问该图片

    <!-- 设置 Servlet 映射-->
    <servlet-mapping>
        <!-- Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <!-- 如访问 img 目录中的 2.jpg 图片 -->
        <url-pattern>/img/2.jpg</url-pattern>
    </servlet-mapping>

模糊匹配 *

使用 * 进行模糊匹配,表示 0 ~ n 个的任意字符

如果通过 http://localhost:8080/servlettest/img/5.jpg 访问该图片不会成功。。。

    <!-- 设置 Servlet 映射-->
    <servlet-mapping>
        <!-- Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <url-pattern>/img/*</url-pattern>
        <url-pattern>*.jpg</url-pattern>
    </servlet-mapping>

默认匹配 /

何为默认匹配,是该路径得不到 “其他 Servlet” 的处理 ,最终会由 <url-pattern>/</url-pattern> 的 Servlet 处理, 如默认的 “default” Servlet 无奈接手 “其他 Servlet” 不处理的请求

访问 http://localhost:8080/servlettest/  ,“其他 Servlet” 不处理,“default Servlet” 会加载首页文件,如果没有文件,显示 “404” 

    <servlet-mapping>
        <!-- Servlet 程序名称 -->
        <servlet-name>TestServlet</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

以下代码来自 Tomcat 中的 conf 目录下的 web.xml 文件,我们可以了解到 Tomcat 自定义了许多 Servlet 处理各种访问

    <!-- The mapping for the default servlet -->
    <servlet-mapping>
        <servlet-name>default</servlet-name>
        <url-pattern>/</url-pattern>
    </servlet-mapping>

    <!-- The mappings for the JSP servlet -->
    <servlet-mapping>
        <servlet-name>jsp</servlet-name>
        <url-pattern>*.jsp</url-pattern>
        <url-pattern>*.jspx</url-pattern>
    </servlet-mapping>

3.4 session-config

设置 Session 超时时间,单位为分钟

    <!-- 设置 Session 超时时间,单位为分钟 -->
    <session-config>
        <session-timeout>30</session-timeout>
    </session-config>

3.5 error-page

设置错误页,当 404 或 500 或 发生异常时可以跳转至指定页面,有三个子标签

  • <error-code></error-code>:用来设置错误码,例如 404, 500等
  • <exception-type></exception-type>:用来设置异常类型,例如 java.lang.NullPointerException(空指针异常)
  • <location></location>:用来设置跳转的错误页
    <!-- 设置错误页 -->
    <error-page>
        <error-code>404</error-code>
        <location>/error_404.html</location>
    </error-page>
    <error-page>
        <exception-type>java.lang.NullPointerException</exception-type>
        <location>/error_null.html</location>
    </error-page>

访问不存在的页面,跳转到错误页进行显示

Tomcat 会自动解析各种 xml 文件,这个过程。。。对于现在的我而言太复杂了

InputSource contextWebXml = this.getContextWebXmlSource();

Tomcat 如何根据 URL 定位到 Servlet

参考自:java - Tomcat 架构原理解析到架构设计借鉴 - 个人文章 - SegmentFault 思否

以 http://user.shopping.com:8080/order/buy 为例,其中 http 为访问协议、user.shopping.com 为主机名、order 可认为是一个工程名、一个 Servlet 对应一个 Wrapper,buy 可认为是一个 Servlet 程序名称

  1. 首先根据协议和端口号确定 Service 和 Engine。Tomcat 默认的 HTTP 连接器监听 8080 端口、默认的 AJP 连接器监听 8009 端口。上面例子中的 URL 访问的是 8080 端口,因此这个请求会被 HTTP 连接器接收,而一个连接器是属于一个 Service 组件的,这样 Service 组件就确定了。我们还知道一个 Service 组件里除了有多个连接器,还有一个容器组件,具体来说就是一个 Engine 容器,因此 Service 确定了也就意味着 Engine 也确定了
  2. 根据域名选定 Host。 Service 和 Engine 确定后,Mapper 组件通过 URL 中的域名去查找相应的 Host 容器,比如例子中的 URL 访问的域名是 user.shopping.com,因此 Mapper 会找到 Host2 这个容器
  3. 根据 URL 路径找到 Context 组件。 Host 确定以后,Mapper 根据 URL 的路径来匹配相应的 Web 应用的路径,比如例子中访问的是 /order,因此找到了 Context4 这个 Context 容器
  4. 根据 URL 路径找到 Wrapper(Servlet)。 Context 确定后,Mapper 再根据 web.xml 中配置的 Servlet 映射路径来找到具体的 Wrapper 和 Servlet

补充,也可以是请求访问 http://user.shopping.com:8080/order/buy.html

5 执行 Servlet

参考文档:

透过现象看本质——什么是servlet_servlet go-CSDN博客

Java servlet执行的完整流程(图解含源码分析)_请简述 servlet 的工作流程?-CSDN博客Tomcat如何创建Servlet? Servlet的执行流程_tomcat先创建servlet还是先创建servletcontext-CSDN博客Java servlet执行的完整流程(图解含源码分析)_请简述 servlet 的工作流程?-CSDN博客

5.1 Servlet 接口(规范)

所有实现 Servlet 接口的类必须实现 init()、Service()、getServletConfig()、getServletInfo()、destroy() 方法

5.2 Servlet 生命周期

  • 执行 Servlet 构造器方法,创建 Servlet 时调用,只调用一次
  • 执行 init 初始化方法,创建 Servlet 时调用,只调用一次
  • 执行 service 方法,每次访问都会调用
  • 执行 destory 方法,JavaWeb 工程停止的时候调用

        服务器找到全类名后,会在缓存中查看是否存在该对象,不存在则通过反射创建对象,同时也创建了 ServletConfig 类,里面存放了一些初始化信息(注意服务器只会创建一次 Servlet 对象,所以 ServletConfig 也只有一个)

5.3 执行 init 初始化方法

原文的解释有点问题,通过代码来说明,根据多态动态绑定,执行 Servlet 子类重写的方法

// 反射创建 Servlet 对象
Servlet obj = Class.forName("全类名").newInstance();

// ServletConfig config
//调用 init 方法初始化
obj.init(config);

// ServletRequest req, ServletResponse resp
// 调用 Service 方法
obj.Service(req, resp);

5.4 执行 service 方法

        仓促抄完,大概就抄这么点,用的时候继承 HttpServlet ,重写 doGet 和 doPost 就好了,忘了写可以通过注解 @WebServlet("/Servlet名称") 配置 Servlet 了,写了注解 @WebServlet,就不要在 web.xml 配置 Servlet,好像会冲突。。。

希望大佬多多指正!!!问题。。。我暂时还找不出来

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1462127.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SparkSQL学习01

目录 1.SparkSQL特点1.1易整合1.2统一的数据访问1.3兼容Hive1.4标准的数据连接 2 SparkSQL编程模型DataFrameDataSet2.1 SQL2.2 DataFrame是什么2.3 DataSet是什么2.4 RDD&#xff0c;DataSet&#xff0c;DataFrame 3 SparkSQL核心编程3.1 编程入口3.2 SparkSQL基本编程3.2.1编…

CQT新里程碑:SOC 2 数据安全认证通过,加强其人工智能支持

Covalent Network&#xff08;CQT&#xff09;发展新里程碑&#xff1a;SOC 2 数据安全认证通过&#xff0c;进一步加强了其人工智能支持 Covalent Network&#xff08;CQT&#xff09;现已完成并通过了严格的 Service Organization Control&#xff08;SOC) 2 Type II 的合规性…

MySQL学习笔记3: MySQL数据库基础

目录 前言目标数据库操作&#xff08;针对database 的操作&#xff09;1. 创建数据库 create database 数据库名;2. 查看数据库 show databases;3. 选中数据库 use 数据库名;4. 删除数据库 drop database 数据库名; mysql中支持的数据类型1. 数值类型: NUMERIC(M,D)2. 字符串类…

如何自定义一个协议

. 如何自定义一个协议 先有一个需求,有个场景,打开外卖软件,会显示商家列表,列表中有很多项,每一项都包含了一些信息,商家的名称,图片,好评率,距离你的位置,评分 这些信息都是通过网络,从服务器获取的, 客户端,需要给服务器发送一个请求,服务器收到请求之后,就给客户端返回一个…

压缩感知中常用的稀疏基

稀疏基的基本概念 在压缩感知中&#xff0c;稀疏基是指可以将信号表示为少数几个非零系数的基。信号在这组基下的表示称为稀疏表示。理想情况下&#xff0c;信号在稀疏基下的大部分系数都应该为零&#xff0c;只有少数几个系数是非零的。稀疏基的选择对信号的稀疏表示和压缩感…

第三十八天| 509. 斐波那契数、70. 爬楼梯、卡码网 爬楼梯、746. 使用最小花费爬楼梯

Leetcode 509. 斐波那契数 题目链接&#xff1a;509 斐波那契数 题干&#xff1a;斐波那契数 &#xff08;通常用 F(n) 表示&#xff09;形成的序列称为 斐波那契数列 。该数列由 0 和 1 开始&#xff0c;后面的每一项数字都是前面两项数字的和。也就是&#xff1a; F(0) 0&a…

【设计模式】01-装饰器模式Decorator

作用&#xff1a;在不修改对象外观和功能的情况下添加或者删除对象功能&#xff0c;即给一个对象动态附加职能 装饰器模式主要包含以下角色。 抽象构件&#xff08;Component&#xff09;角色&#xff1a;定义一个抽象接口以规范准备接收附加责任的对象。具体构件&#xff08…

opengl 学习着色器

一.GLSL 着色器是使用一种叫GLSL的类C语言写成的。GLSL着色器编码顺序&#xff1a;声明版本》定义输入输出》uniform》main函数。每个着色器的入口点是main函数&#xff0c;在main函数中我们处理所有的输入变量&#xff0c;并将结果输出到输出变量中。如下图&#xff1a; #ver…

Nginx----高性能的WEB服务端

一、Nginx介绍 1、什么是Nginx Nginx Nginx是一个高性能的HTTP和反向代理服务器。是一款轻量级的高性能的web服务器/反向代理服务器/电子邮件&#xff08;IMAP/POP3&#xff09;代理服务器&#xff0c;单台物理服务器可支持30 000&#xff5e;50 000个并发请求。 一款高性能…

【广度优先搜索】【网格】【割点】【 推荐】1263. 推箱子

作者推荐 视频算法专题 涉及知识点 广度优先搜索 网格 割点 并集查找 LeetCode:1263. 推箱子 「推箱子」是一款风靡全球的益智小游戏&#xff0c;玩家需要将箱子推到仓库中的目标位置。 游戏地图用大小为 m x n 的网格 grid 表示&#xff0c;其中每个元素可以是墙、地板或…

项目中使用Echarts图表treemap制作

Examples - Apache ECharts 页面结构 <template><div class"echart-chart" :id"id"></div> </template> <script> import { getOption } from ./chartOption; export default {name: TreeMapChart,props: [id, data, type]…

LaWGPT—基于中文法律知识的大模型

文章目录 LaWGPT&#xff1a;基于中文法律知识的大语言模型数据构建模型及训练步骤两个阶段二次训练流程指令精调步骤计算资源 项目结构模型部署及推理 LawGPT_zh&#xff1a;中文法律大模型&#xff08;獬豸&#xff09;数据构建知识问答模型推理训练步骤 LaWGPT&#xff1a;基…

红蓝对抗:网络安全领域的模拟实战演练

引言&#xff1a; 随着信息技术的快速发展&#xff0c;网络安全问题日益突出。为了应对这一挑战&#xff0c;企业和组织需要不断提升自身的安全防护能力。红蓝对抗作为一种模拟实战演练方法&#xff0c;在网络安全领域得到了广泛应用。本文将介绍红蓝对抗的概念、目的、过程和…

微服务篇之监控

一、为什么要监控 1.问题定位 假设客户端查询一些东西的时候&#xff0c;需要经过网关&#xff0c;然后服务A调用服务H&#xff0c;服务H调用K&#xff0c;服务K调用MySQL&#xff0c;当查询不出来的时候&#xff0c;我们不能快速定位到底是哪个服务的问题&#xff0c;这就需要…

量子加密机的工作原理是什么

量子加密机&#xff0c;作为现代加密技术的一大飞跃&#xff0c;正逐渐成为信息安全领域的研究热点。与传统的加密方法相比&#xff0c;量子加密技术以其独特的优势&#xff0c;为信息安全提供了更为坚实的保障。 量子加密的核心在于利用量子力学的特性&#xff0c;尤其是量子纠…

uni-app 开发调试自动打开手机屏幕大小界面(Aidex移动端开发项目)

上效果&#xff1a; 下载Aidex的移动端项目并打开&#xff1a; 若依-ruoyi-AiDex-Uniapp: 若依-Ruoyi APP 移动解决方案&#xff0c;基于uniappuView封装的一套基础模版&#xff0c;开箱即用&#xff0c;免费开源&#xff0c;一份代码多终端适配&#xff0c;支持H5、支付宝小程…

Python hasattr函数

在Python编程中&#xff0c;hasattr()函数是一个非常有用的内置函数之一&#xff0c;用于检查对象是否具有指定的属性或方法。这个函数能够帮助我们在运行时动态地检查对象的属性和方法&#xff0c;从而避免由于缺少属性或方法而导致的异常。本文将深入探讨Python中的hasattr()…

unity学习(36)——角色选取界面(自制美工)

1.添加一个背景图片&#xff0c;记不住可以查之前的资料&#xff08;4&#xff09; 图片拖入asset&#xff0c;属性设成sprite&#xff1b;把图片拖到source image中&#xff1b;colour白色&#xff08;透明&#xff0c;点一下右边的笔即可&#xff09;&#xff1b;material为…

《Python 语音转换简易速速上手小册》第2章 Python 编程基础(2024 最新版)

文章目录 2.1 Python 语言基础2.1.1 基础知识深入基础总结 2.1.2 主要案例&#xff1a;数据分析脚本案例介绍案例 Demo案例分析 2.1.3 扩展案例 1&#xff1a;自动化邮件发送案例介绍案例 Demo案例分析 2.1.4 扩展案例 2&#xff1a;网页数据抓取案例介绍案例 Demo案例分析 2.2…

fastApi笔记05-路径参数和数值校验

使用Path可以对路径参数声明与Query相同类型的校验和元数据 from typing import Annotatedfrom fastapi import FastAPI, Path, Queryapp FastAPI()app.get("/items/{item_id}") async def read_items(item_id: Annotated[int, Path(title"The ID of the item …