当前位置: 首页 > news >正文

从0搭建Tomcat第二天:深入理解Servlet容器与反射机制

在上一篇博客中,我们从0开始搭建了一个简易的Tomcat服务器,并实现了基本的HTTP请求处理。今天,我们将继续深入探讨Tomcat的核心组件之一——Servlet容器,并介绍如何使用反射机制动态加载和管理Servlet。

1. Servlet容器的作用

Servlet容器是Tomcat的核心组件之一,负责管理Servlet的生命周期,并将HTTP请求分发给相应的Servlet进行处理。Servlet容器的主要功能包括:

  • 加载和初始化Servlet:在Tomcat启动时,Servlet容器会加载所有的Servlet类,并调用它们的init方法进行初始化。

  • 处理HTTP请求:当客户端发送HTTP请求时,Servlet容器会根据请求的URL找到对应的Servlet,并调用其service方法处理请求。

  • 销毁Servlet:在Tomcat关闭时,Servlet容器会调用Servlet的destroy方法进行资源释放。

2. 实现Servlet容器

2.1 使用Map管理Servlet

我们可以使用一个Map来管理Servlet对象,其中key是Servlet的访问路径,value是对应的Servlet对象。以下是一个简单的实现:

Map<String, Servlet> servletContainer = new HashMap<>();// 在Tomcat启动时加载Servlet
servletContainer.put("/login", new LoginServlet());
servletContainer.put("/show", new ShowServlet());// 根据请求路径获取相应的Servlet
Servlet servlet = servletContainer.get(request.getPath());
if (servlet != null) {servlet.service(request, response);
}

在这个实现中,我们手动将LoginServletShowServlet对象放入servletContainer中。然而,在实际的Tomcat中,Servlet的加载和管理是自动化的,通常通过反射机制来实现。

2.2 使用反射动态加载Servlet

为了动态加载Servlet,我们可以使用Java的反射机制。反射允许我们在运行时获取类的信息,并动态创建对象。以下是一个使用反射加载Servlet的示例:

package com.qcby.Util;import java.io.File;
import java.util.ArrayList;
import java.util.List;public class SearchClassUtil {public static List<String> classPaths = new ArrayList<String>();/*** 扫描指定包下的所有类,并获取全路径名*/public static List<String> searchClass() {String basePack = "com.qcby.webapps";String classPath = SearchClassUtil.class.getResource("/").getPath();basePack = basePack.replace(".", File.separator);String searchPath = classPath + basePack;doPath(new File(searchPath), classPath);return classPaths;}/*** 递归扫描目录,获取所有类的全路径名*/private static void doPath(File file, String classpath) {if (file.isDirectory()) {File[] files = file.listFiles();if (files != null) {for (File f1 : files) {doPath(f1, classpath);}}} else {if (file.getName().endsWith(".class")) {String path = file.getPath().replace(classpath.replace("/", "\\").replaceFirst("\\\\", ""), "").replace("\\", ".").replace(".class", "");classPaths.add(path);}}}/*** 根据全路径名加载类对象*/public static List<Class<?>> loadClasses(List<String> classPaths) {List<Class<?>> classes = new ArrayList<>();for (String className : classPaths) {try {Class<?> clazz = Class.forName(className);classes.add(clazz);} catch (ClassNotFoundException e) {System.err.println("无法加载类: " + className);e.printStackTrace();}}return classes;}/*** 获取类中的 @WebServlet 注解值*/public static void getWebServletAnnotation(List<Class<?>> classes) {for (Class<?> clazz : classes) {if (clazz.isAnnotationPresent(WebServlet.class)) {WebServlet webServletAnnotation = clazz.getAnnotation(WebServlet.class);String urlMapping = webServletAnnotation.urlMapping();System.out.println("类 " + clazz.getName() + " 的 @WebServlet 注解值: " + urlMapping);}}}public static void main(String[] args) {List<String> classPaths = searchClass();List<Class<?>> classes = loadClasses(classPaths);getWebServletAnnotation(classes);}
}

在这个代码中,我们使用SearchClassUtil类扫描指定包下的所有类,并加载带有@WebServlet注解的类。通过反射,我们可以动态获取这些类的注解信息,并根据注解中的urlMapping值将Servlet对象放入servletContainer中。

2.3 自定义@WebServlet注解

为了标识哪些类是Servlet,我们可以定义一个自定义注解@WebServlet

package com.qcby.Util;import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.TYPE)
public @interface WebServlet {String urlMapping() default "";
}

在Servlet类中,我们可以使用这个注解来指定Servlet的访问路径:

package com.qcby.webapps.myweb;import com.qcby.Util.WebServlet;
import com.qcby.servlet.HttpServlet;
import com.qcby.servlet.req.HttpServletRequest;
import com.qcby.servlet.req.HttpServletResponse;@WebServlet(urlMapping = "/login")
public class LoginServlet extends HttpServlet {@Overrideprotected void doGet(HttpServletRequest request, HttpServletResponse response) {System.out.println("处理登录的GET请求");}@Overrideprotected void doPost(HttpServletRequest request, HttpServletResponse response) {System.out.println("处理登录的POST请求");}
}

3. 总结

通过本文,我们深入探讨了Tomcat的Servlet容器,并实现了动态加载和管理Servlet的功能。我们使用反射机制扫描指定包下的类,并根据自定义注解@WebServlet动态创建Servlet对象。这种方式使得Servlet的管理更加灵活和自动化。

在下一篇博客中,我们将继续探讨Tomcat的其他核心组件,如线程池、连接器等。如果你对Tomcat的更多细节感兴趣,欢迎继续关注。


参考文献:

  • Java Reflection Tutorial

  • Apache Tomcat Documentation

相关推荐:

  • 深入理解Java Web开发

  • Tomcat源码解析

相关文章:

从0搭建Tomcat第二天:深入理解Servlet容器与反射机制

在上一篇博客中&#xff0c;我们从0开始搭建了一个简易的Tomcat服务器&#xff0c;并实现了基本的HTTP请求处理。今天&#xff0c;我们将继续深入探讨Tomcat的核心组件之一——Servlet容器&#xff0c;并介绍如何使用反射机制动态加载和管理Servlet。 1. Servlet容器的作用 S…...

【Python】yield函数

【Python】yield函数 1. yield介绍2.yield基本用法3.yield高级用法3.1 yield send() 方法3.2 yield from方法3.3 yield 和yield from叠加处理复杂情况下的叠加 4.yield主要应用场景5.总结 python官方api地址 1. yield介绍 在Python中&#xff0c;yield关键字主要用于生成器函…...

Android13修改多媒体默认音量

干就完了! 设置音量为最大音量,修改如下: /framework/base/media/java/android/media/AudioSystem.java/** hide */public static int[] DEFAULT_STREAM_VOLUME new int[] {4, // STREAM_VOICE_CALL7, // STREAM_SYSTEM5, // STREAM_RING-5, // STREAM_MUSIC15, // STREAM…...

nginx+keepalived负载均衡及高可用

一、环境准备 主机名ip地址备注openEuler-1 192.168.121.11&#xff08;本机&#xff09; 192.168.131.11&#xff08;心跳连接&#xff09; nginx主负载均衡调度器openEuler-2 192.168.121.12&#xff08;本机&#xff09; 192.168.131.12&#xff08;心跳连接&#xff09; n…...

SP导入智能材质球

智能材质球路径 ...\Adobe Substance 3D Painter\resources\starter_assets\smart-materials 放入之后就会自动刷新...

Kotlin语言特性(一):空安全、扩展函数与协程

Kotlin语言特性&#xff08;一&#xff09;&#xff1a;空安全、扩展函数与协程 一、引言 Kotlin作为Android官方推荐的开发语言&#xff0c;相比Java具有诸多现代化特性。本文将重点介绍Kotlin三个最具特色的语言特性&#xff1a;空安全、扩展函数和协程&#xff0c;并结合A…...

Sqlserver安全篇之_启用TLS即配置SQL Server 数据库引擎以加密连接

官方文档 https://learn.microsoft.com/zh-cn/sql/database-engine/configure-windows/configure-sql-server-encryption?viewsql-server-ver16 https://learn.microsoft.com/zh-cn/sql/database-engine/configure-windows/manage-certificates?viewsql-server-ver15&pre…...

Python 爬虫 – BeautifulSoup

Python 爬虫&#xff08;Web Scraping&#xff09;是指通过编写 Python 程序从互联网上自动提取信息的过程。 爬虫的基本流程通常包括发送 HTTP 请求获取网页内容、解析网页并提取数据&#xff0c;然后存储数据。 Python 的丰富生态使其成为开发爬虫的热门语言&#xff0c;特…...

【星云 Orbit-STM32F4】07. 用判断数据尾来接收据的串口通用程序框架

【星云 Orbit-STM32F4】用判断数据尾来接收一串数据的串口通用程序框架 摘要 本文介绍了一种基于STM32F407微控制器的串口数据接收通用程序框架。该框架通过判断数据尾来实现一串数据的完整接收&#xff0c;适用于需要可靠数据传输的应用场景。本文从零开始&#xff0c;详细讲…...

授权与认证之jwt(一)创建Jwt工具类

JWT的Token要经过加密才能返回给客户端&#xff0c;包括客户端上传的Tokn,后端项目需要验证核 实。于是我们需要一个WT工具类&#xff0c;用来加密Token和验证Token的有效性。 一、导入依赖 <dependency><groupId>com.auth0</groupId><artifactId>jav…...

Kubernetes Service服务发现dns之CoreDNS

文章目录 背景什么是Service、服务发现、Endpoint什么是CoreDNSCoreDNS 的工作原理 常用命令coredns 运行状态根据服务名&#xff0c;判断某个服务dns解析是否正常 背景 Kubernetes 集群内部的服务发现是微服务架构的核心基础&#xff0c;而 DNS 服务则是实现这一机制的关键组…...

Spring Boot 测试:单元、集成与契约测试全解析

一、Spring Boot 分层测试策略 Spring Boot 应用采用经典的分层架构&#xff0c;不同层级的功能模块对应不同的测试策略&#xff0c;以确保代码质量和系统稳定性。 Spring Boot 分层架构&#xff1a; Spring Boot分层架构 A[客户端] -->|HTTP 请求| B[Controller 层] …...

用友NC系列漏洞检测利用工具

声明&#xff01;本文章所有的工具分享仅仅只是供大家学习交流为主&#xff0c;切勿用于非法用途&#xff0c;如有任何触犯法律的行为&#xff0c;均与本人及团队无关&#xff01;&#xff01;&#xff01; 目录标题 YongYouNcTool启动及适配环境核心功能界面预览一键检测命令执…...

PostgreSQL 创建表格

PostgreSQL 创建表格 在数据库管理中&#xff0c;表格&#xff08;Table&#xff09;是数据存储的基础。PostgreSQL作为一款强大的开源对象关系型数据库管理系统&#xff08;ORDBMS&#xff09;&#xff0c;创建表格是其最基本的功能之一。本文将详细讲解如何在PostgreSQL中创…...

一周一个Unity小游戏2D反弹球游戏 - 球的死区及球重生

前言 本文将实现当球弹到球板下方的死亡区域后,球会被重置到球板上发射点,并且重置物理状态的逻辑。 创建球的死亡区 之前创建的在屏幕下方的空气墙碰撞体可以将其Is Trigger勾选上,让其成为一个触发器,用来检测球是否进入该区域,如下。 创建一个脚本名为Deadzone…...

本地部署 DeepSeek:从 Ollama 配置到 Spring Boot 集成

前言 随着人工智能技术的迅猛发展&#xff0c;越来越多的开发者希望在本地环境中部署和调用 AI 模型&#xff0c;以满足特定的业务需求。本文将详细介绍如何在本地环境中使用 Ollama 配置 DeepSeek 模型&#xff0c;并在 IntelliJ IDEA 中创建一个 Spring Boot 项目来调用该模型…...

vue3:三项目增加404页面

一、路由添加 1、官网地址 带参数的动态路由匹配 | Vue Routerhttps://router.vuejs.org/zh/guide/essentials/dynamic-matching.html 2、复制核心语句 { path: /:pathMatch(.*)*, name: NotFound, component: NotFound } 3、粘贴到路由index.js中 4、建立页面 在view文件夹…...

MCAL(Microcontroller Abstraction Layer)介绍

目录 MCAL的核心作用 MCAL的模块组成 1. 微控制器驱动&#xff08;Microcontroller Drivers&#xff09; 2. I/O驱动&#xff08;DIO, PWM, ADC等&#xff09; 3. 通信驱动&#xff08;Communication Drivers&#xff09; 4. 存储驱动&#xff08;Memory Drivers&#xf…...

爬虫:PhantomJS的详细使用和实战案例

文章目录 一、PhantomJS介绍1.1 什么是 PhantomJS1.2 PhantomJS 的特点与优势二、PhantomJS 的安装2.1 在 macOS 上安装 PhantomJS2.2 在 Linux 上安装 PhantomJS2.3 在 Windows 上安装 PhantomJS2.4 验证安装三、PhantomJS 的基本使用3.1 示例 1:打开网页并截图3.2 示例 2:获…...

目标检测——数据处理

1. Mosaic 数据增强 Mosaic 数据增强步骤: (1). 选择四个图像&#xff1a; 从数据集中随机选择四张图像。这四张图像是用来组合成一个新图像的基础。 (2) 确定拼接位置&#xff1a; 设计一个新的画布(输入size的2倍)&#xff0c;在指定范围内找出一个随机点&#xff08;如…...

3步攻克Linux应用管理痛点:面向开发者的AppImageLauncher优化方案

3步攻克Linux应用管理痛点&#xff1a;面向开发者的AppImageLauncher优化方案 【免费下载链接】AppImageLauncher Helper application for Linux distributions serving as a kind of "entry point" for running and integrating AppImages 项目地址: https://gitc…...

Citra模拟器终极指南:免费畅玩3DS游戏的完整教程

Citra模拟器终极指南&#xff1a;免费畅玩3DS游戏的完整教程 【免费下载链接】citra A Nintendo 3DS Emulator 项目地址: https://gitcode.com/gh_mirrors/cit/citra 任天堂3DS模拟器Citra是一款开源的高性能游戏模拟工具&#xff0c;让PC用户能够流畅体验《精灵宝可梦》…...

Navicat重置工具终极指南:macOS用户免费解锁专业功能的完整方案

Navicat重置工具终极指南&#xff1a;macOS用户免费解锁专业功能的完整方案 【免费下载链接】navicat_reset_mac navicat mac版无限重置试用期脚本 Navicat Mac Version Unlimited Trial Reset Script 项目地址: https://gitcode.com/gh_mirrors/na/navicat_reset_mac 还…...

PyTorch 2.8镜像行业落地:教育机构AI教学平台+视频课件自动生成实践

PyTorch 2.8镜像行业落地&#xff1a;教育机构AI教学平台视频课件自动生成实践 1. 教育行业AI转型的机遇与挑战 教育行业正经历数字化转型浪潮&#xff0c;传统教学方式面临三大核心痛点&#xff1a; 内容生产效率低&#xff1a;教师手工制作课件平均耗时3-5小时/课时个性化…...

Swift-All部署教程:快速搭建多模型推理与微调环境

Swift-All部署教程&#xff1a;快速搭建多模型推理与微调环境 1. 从零开始&#xff1a;为什么你需要Swift-All&#xff1f; 如果你正在研究大模型&#xff0c;或者想把大模型用在实际项目里&#xff0c;大概率会遇到这几个头疼的问题&#xff1a; 模型太多&#xff0c;下载太…...

5分钟搞定!OpenCode+Qwen3-4B本地AI编程助手一键部署教程

5分钟搞定&#xff01;OpenCodeQwen3-4B本地AI编程助手一键部署教程 1. 引言&#xff1a;为什么你需要一个本地AI编程助手&#xff1f; 想象一下这个场景&#xff1a;你正在开发一个核心功能模块&#xff0c;需要快速生成一段复杂的业务逻辑代码。你打开浏览器&#xff0c;准…...

ESTree节点遍历终极指南:深度优先与广度优先算法完整解析

ESTree节点遍历终极指南&#xff1a;深度优先与广度优先算法完整解析 【免费下载链接】estree The ESTree Spec 项目地址: https://gitcode.com/gh_mirrors/es/estree JavaScript开发者们&#xff0c;你们是否在构建代码分析工具时遇到过AST遍历的难题&#xff1f;&…...

Janus-Pro-7B在CNN图像识别中的增强应用

Janus-Pro-7B在CNN图像识别中的增强应用 1. 引言 图像识别技术正在经历一场革命性的变革。传统的CNN模型虽然在图像分类任务上表现出色&#xff0c;但在复杂场景和多模态理解方面仍存在局限。今天我们要介绍的Janus-Pro-7B&#xff0c;作为一个统一的多模态理解和生成框架&am…...

别再用PS硬P了!用Python+OpenCV实现泊松融合,5分钟搞定图片无缝拼接

告别PS繁琐操作&#xff1a;5行Python代码实现专业级图片融合 每次在Photoshop里手动调整图层蒙版、反复擦除边缘时&#xff0c;你是否想过——数字图像处理应该更智能&#xff1f;2023年&#xff0c;我们完全可以用代码自动化完成这些重复劳动。本文将带你用PythonOpenCV实现泊…...

5分钟搞定!FLUX.2-Klein-9B在ComfyUI中的快速部署与初体验

5分钟搞定&#xff01;FLUX.2-Klein-9B在ComfyUI中的快速部署与初体验 1. 为什么选择FLUX.2-Klein-9B 如果你正在寻找一个既能高质量生成图像&#xff0c;又对中文提示词理解优秀的AI模型&#xff0c;FLUX.2-Klein-9B值得一试。这个模型特别适合需要频繁进行图像编辑的场景&a…...