搭建Tomcat(四)---Servlet容器

目录

引入

Servlet容器

一、优化MyTomcat

①先将MyTomcat的main函数搬过来:

②将getClass()函数搬过来

 ③创建容器

 ④连接ServletConfigMapping和MyTomcat

连接:

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping

b.MyTomcat

二、优化Server

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

方法二: 直接嵌套

三、获取类对象

处理请求:

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

结果:


引入

在先前的tomcat搭建学习中,已经对tomcat的雏形做了基本的实现,即如下的过程:

接下来继续tomcat的搭建。

Servlet容器

目前Servlet容器:

 接下来,我们来优化一下MyTomcat:

一、优化MyTomcat

首先在tomcat包下创建一下config包,然后在里面创建文件ServletConfigMapping:

那么ServletConfigMapping里面编写什么呢?就是从MyTomcat中编写的内容,现在需要copy到这个文件中,实现MyTomcat的解放:

【ServletConfigMapping在tomcat中就是为了获取配置信息;而MyTomcat中写的,正是扫描的过程,即未来扫描动态资源映射表的过程,所以现在将它放进合适的地方,即ServletConfigMapping里面】

注意: 不是原封不动的照搬,需要做一些小小的调整。

①先将MyTomcat的main函数搬过来:

注意这里做了个小调整,将MyTomcat的main函数里面的内容放进了代码块中;

static代码块的特点:在main函数执行之前先执行

static{try {// 1. 扫描包路径 (com.wzh.tomcat.myweb)String packageName = "com.qcby.tomcat.myweb";List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中// 2. 遍历所有类,检查是否有@WebServlet注解for (Class<?> clazz : classes) {if (clazz.isAnnotationPresent(WebServlet.class)) {// 3. 获取@WebServlet注解的值WebServlet webServlet = clazz.getAnnotation(WebServlet.class);System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);}}} catch (Exception e) {e.printStackTrace();}}

②将getClass()函数搬过来

 这里直接copy就好,放到刚才的static代码块之下:

/*** 获取指定包下的所有类** @param packageName 包名,例如 "com.qcby.tomcat.myweb"* @return 类对象列表* @throws Exception*///将MyTomcat中的getClass方法也粘贴过来private static List<Class<?>> getClasses(String packageName) throws Exception {List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中String path = packageName.replace('.', '/'); // 将包名转换为文件路径// 通过类加载器获取包的资源路径ClassLoader classLoader = Thread.currentThread().getContextClassLoader();Enumeration<URL> resources = classLoader.getResources(path);while (resources.hasMoreElements()) {URL resource = resources.nextElement();File directory = new File(resource.toURI());// 扫描文件夹下的所有类文件if (directory.exists()) {for (File file : directory.listFiles()) {if (file.getName().endsWith(".class")) {    //获得.class文件// 获取类的完整类名String className = packageName + "." + file.getName().replace(".class", "");classes.add(Class.forName(className));}}}}return classes;}

 基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可:

public static void main(String[] args) {//当然也可以直接将static中的内容放进main函数里面(不常用)}

 ③创建容器

真正的servlet容器:

public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();

/*
* 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态
* 多态--父类的引用指向子类的对象
* 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象
* 这些类对象都继承了HttpServlet
* 即:
* MyFirstServlet的类对象--->是HttpServlet的子类
* MySecondServlet的类对象--->是HttpServlet的子类
* MyThirdServlet的类对象--->是HttpServlet的子类
*【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】
* */

 ④连接ServletConfigMapping和MyTomcat

我们只是想优化MyTomcat,而不是彻底换掉MyTomcat,所以现在我们要做的就是将创建的ServletConfigMapping和先前的MyTomcat连接起来:

当然由于主要代码已经放进ServletConfigMapping中了,所以原先的MyTomcat可以清空了。

连接:

注意②中的执行方法---放main;那么想要连接ServletConfigMapping和MyTomcat,我们依旧可以采用这种方法:让MyTomcat调用ServletConfigMapping中的方法,迫使ServletConfigMapping中的static代码块运行。

public class MyTomcat {public static void main(String[] args) {ServletConfigMapping.init();//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接}
}

当然这个写在ServletConfigMapping中的方法,可以是空的。

⑤完整的ServletConfigMapping和MyTomcat方法:

a.ServletConfigMapping
package com.qcby.tomcat.config;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.webservlet.WebServlet;import java.io.File;
import java.net.URL;
import java.util.*;/*
* servlet容器
* */public class ServletConfigMapping {//真正的servlet容器/** 至于这里value的位置为什么写HttpServlet呢?按理说应该写类对象----多态* 多态--父类的引用指向子类的对象* 不妨来看一下,能写进来的MyFirstServlet、MySecondServlet、MyThirdServlet等的类对象* 这些类对象都继承了HttpServlet* 即:* MyFirstServlet的类对象--->是HttpServlet的子类* MySecondServlet的类对象--->是HttpServlet的子类* MyThirdServlet的类对象--->是HttpServlet的子类*【多态性--父类的引用指向子类的对象,并且子类的对象可以向上转型为父类的引用】* */public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();//扫描遍历(即MyTomcat之中的内容)//static代码块在main方法执行之前执行//将MyTomcat中的main方法放进代码块中static{try {// 1. 扫描包路径 (com.wzh.tomcat.myweb)String packageName = "com.qcby.tomcat.myweb";List<Class<?>> classes = getClasses(packageName);   //通过getClasses()方法获取到了myweb这个包下面的所有类的类对象,并将其放到了类对象中// 2. 遍历所有类,检查是否有@WebServlet注解for (Class<?> clazz : classes) {if (clazz.isAnnotationPresent(WebServlet.class)) {// 3. 获取@WebServlet注解的值WebServlet webServlet = clazz.getAnnotation(WebServlet.class);System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.url());classMap.put(webServlet.url(),(Class<HttpServlet>) clazz);}}} catch (Exception e) {e.printStackTrace();}}/*** 获取指定包下的所有类** @param packageName 包名,例如 "com.qcby.tomcat.myweb"* @return 类对象列表* @throws Exception*///将MyTomcat中的getClass方法也粘贴过来private static List<Class<?>> getClasses(String packageName) throws Exception {List<Class<?>> classes = new ArrayList<>(); //将类文件封装进List中String path = packageName.replace('.', '/'); // 将包名转换为文件路径// 通过类加载器获取包的资源路径ClassLoader classLoader = Thread.currentThread().getContextClassLoader();Enumeration<URL> resources = classLoader.getResources(path);while (resources.hasMoreElements()) {URL resource = resources.nextElement();File directory = new File(resource.toURI());// 扫描文件夹下的所有类文件if (directory.exists()) {for (File file : directory.listFiles()) {if (file.getName().endsWith(".class")) {    //获得.class文件// 获取类的完整类名String className = packageName + "." + file.getName().replace(".class", "");classes.add(Class.forName(className));}}}}return classes;}//基于static代码块会在main方法之前执行的特性,想要执行static中的内容,只需要放一个空的main函数运行即可public static void main(String[] args) {//当然也可以直接将static中的内容放进main函数里面(不常用)}public static void init(){}}
b.MyTomcat
package com.qcby.tomcat;import com.qcby.tomcat.config.ServletConfigMapping;public class MyTomcat {public static void main(String[] args) {ServletConfigMapping.init();//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接}
}

二、优化Server

用同样的方法,首先把server的main函数解放掉:

【将main函数中的内容取出,并放进新创的serverInit函数中,删除原来的main函数即可。】

//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}

 方法1:调用(可以用,但是这里咱们不用,过于冗余)

接着用同样的操作,在MyTomcat中调用这新创建的方法:

package com.qcby.tomcat;import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;public class MyTomcat {public static void main(String[] args) {try {//通过调用ServletConfigMapping中的静态方法来促使ServletConfigMapping类的执行,即将这两个类通过这种方式连接ServletConfigMapping.init();//初始化servlet容器Server.serverInit();//启动server服务} catch (Exception e) {e.printStackTrace();}}
}

【看着代码变多了很多,实际上就加了一行,然后为了安全性,抛出了异常】

方法二: 直接嵌套

这里我们用另一种方法去优化Server,即将Server和MyTomcat连接起来:

即直接将方才做过修改的代码粘贴到MyTomcat中,并且在MyTomcat的main函数中调用Server的serverInit(即过去Server类的main函数)

如下是这一小阶段完成后的MyTomcat代码示意:

package com.qcby.tomcat;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;public class MyTomcat {//实例化Requestpublic static Request request = new Request();public static void main(String[] args) throws Exception {ServletConfigMapping.init();serverInit();}//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}public static void handler(Socket socket) throws Exception {//读取请求的数据InputStream inputStream = socket.getInputStream();requestContext(inputStream);}public static void requestContext(InputStream inputStream) throws IOException { //获取全部信息//将bit流转为文字信息int count = 0;while (count == 0) {count = inputStream.available();}byte[] bytes = new byte[count];inputStream.read(bytes);String Context = new String(bytes);System.out.println(Context);//解析数据if (Context.equals("")) {System.out.println("你输入了一个空请求");} else {String firstLine = Context.split("\\n")[0];String method = firstLine.split("\\s")[0];String path = firstLine.split("\\s")[1];System.out.println(method + " " + path);//任何请求都会被打到这个类中,随后就会被解析//将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)request.setMethod(method);request.setPath(path);}}
}

三、获取类对象

前面已经对MyTomcat做了优化,下一步就是去获取类对象;

如何获取类对象呢?简单来说谁存储着类对象的信息呢?---Request

那么就要接着对server类做出优化---因为目前Server和MyTomcat类还是存在问题,两者打不通;

既然想把两者联系起来,那么就要有嵌套调用或是直接嵌套。

在这里,我们依旧是用直接嵌套的方法写:

处理请求:

处理接收到的请求,首先在Map映射中寻找是否能够匹配上key值,即传送的path值

能查到则取获取Map的value值,即类对象: 

 //处理请求public static void dis(Request request) throws Exception {if(!request.getPath().equals("")){if(ServletConfigMapping.classMap.get(request.getPath())!=null){//能进到这里,则说明能够获取Map中对应的key值,能够匹配上Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)servlet.doGet(request,response);}}}

上面的代码最终是要被requestContext(InputStream inputStream)函数调用的。

即最终的MyTomcat代码:

package com.qcby.tomcat;import com.qcby.tomcat.HttpServlet.HttpServlet;
import com.qcby.tomcat.Request.Request;
import com.qcby.tomcat.Response.Response;
import com.qcby.tomcat.config.ServletConfigMapping;
import com.qcby.tomcat.socket.Server;import java.io.IOException;
import java.io.InputStream;
import java.net.ServerSocket;
import java.net.Socket;public class MyTomcat {//实例化Requestpublic static Request request = new Request();public static Response response = new Response();public static void main(String[] args) throws Exception {//两项调用连接ServletConfigMapping.init();serverInit();}//处理请求public static void dis(Request request) throws Exception {if(!request.getPath().equals("")){if(ServletConfigMapping.classMap.get(request.getPath())!=null){//能进到这里,则说明能够获取Map中对应的key值,能够匹配上Class<HttpServlet> classServlet=ServletConfigMapping.classMap.get(request.getPath());HttpServlet servlet=classServlet.newInstance(); //newInstance动态地创建一个类的新实例(对象)servlet.doGet(request,response);}}}//优化Server,取出main方法public static void serverInit() throws Exception {// 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信ServerSocket serverSocket = new ServerSocket(8080);System.out.println("****************server start.....");//2.接受请求数据while (true) {Socket socket = serverSocket.accept();  //--------------------->注意:此时监听网卡的是:主线程System.out.println("有客户进行了链接");new Thread(() -> {//处理数据---------》数据的处理在于读和写try {handler(socket);} catch (Exception e) {e.printStackTrace();}}).start();}}public static void handler(Socket socket) throws Exception {//读取请求的数据InputStream inputStream = socket.getInputStream();requestContext(inputStream);}public static void requestContext(InputStream inputStream) throws Exception { //获取全部信息//将bit流转为文字信息int count = 0;while (count == 0) {count = inputStream.available();}byte[] bytes = new byte[count];inputStream.read(bytes);String Context = new String(bytes);System.out.println(Context);//解析数据if (Context.equals("")) {System.out.println("你输入了一个空请求");} else {String firstLine = Context.split("\\n")[0];String method = firstLine.split("\\s")[0];String path = firstLine.split("\\s")[1];System.out.println(method + " " + path);//任何请求都会被打到这个类中,随后就会被解析//将解析后的数据(method和path放进申请的static的Request实例中--再被运输给其他需要的地方)request.setMethod(method);request.setPath(path);}//调用上面的处理函数dis(request);}
}

实现上述操作后,就可以测试是否正确:

本地的Key值是 “/myFrist”

去客户端(浏览器)输入试试:

 输入对应key值的:

结果:

取看输出这里:
 可以看出,上述代码被顺利执行了。

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

构建一个rust生产应用读书笔记四(实战3)

从这一节开始&#xff0c;我们将继续完善邮件订阅生产级应用&#xff0c;根据作者的选型sqlx作为数据库操作的类库&#xff0c;它有如下优点&#xff1a; 它旨在提供高效、安全且易于使用的数据库交互体验。sqlx 支持多种数据库&#xff0c;包括 PostgreSQL、MySQL 和 SQLite&…

网络安全-------防止被抓包

1.Ios应用网络安全之https 安全套接字层 (Secure Socket Layer, SSL) 是用来实现互联网安全通信的最普遍的标准。Web 应用程序使用 HTTPS&#xff08;基于 SSL 的 HTTP&#xff09;&#xff0c;HTTPS 使用数字证书来确保在服务器和客户端之间进行安全、加密的通信。在 SSL 连接…

WebSocket 与 Server-Sent Events (SSE) 的对比与应用

目录 ✨WebSocket&#xff1a;全双工通信的利器&#x1f4cc;什么是 WebSocket&#xff1f;&#x1f4cc;WebSocket 的特点&#x1f4cc;WebSocket 的优点&#x1f4cc;WebSocket 的缺点&#x1f4cc;WebSocket 的适用场景 ✨Server-Sent Events (SSE)&#xff1a;单向推送的轻…

CAD c# 生成略缩图预览

代码如下&#xff1a; using (Transaction tr currentdb.TransactionManager.StartTransaction()){//当前数据库开启事务using (Database tempdb new Database(false, true)) //创建临时数据库(两个参数&#xff1a;是否创建符号表&#xff0c;不与当前文档关联){try{Bitmap …

娱乐五子棋(附加源码)

一写在开头 上期代码主要实现瀑布流功能&#xff0c;本期就来实现五子棋小游戏&#xff0c;开发久了很多功能都是通过框架组件库来完成&#xff0c;但是如果组件满足不了开发需求&#xff0c;还需要开发人员手动封装组件&#xff0c;专门出这样一期文章&#xff0c;通过原生js实…

XMOS将在CES 2025上展出多款由边缘AI驱动的创新音效、音频、识别和处理解决方案

全球智能物联网技术领导者暨匠心独到的半导体科技企业XMOS宣布&#xff1a;该公司将再次参加2025年国际消费电子展&#xff08;CES 2025&#xff09;&#xff0c;并将在本届CES上展出一系列由人工智能&#xff08;AI&#xff09;驱动的全新空间音效、语音捕获与降噪、音视频多模…

HCIA-Access V2.5_2_2_2网络通信基础_IP编址与路由

网络层数据封装 首先IP地址封装在网络层&#xff0c;它用于标识一台网络设备&#xff0c;其中IP地址分为两个部分&#xff0c;网络地址和主机地址&#xff0c;通过我们采用点分十进制的形式进行表示。 IP地址分类 对IP地址而言&#xff0c;它细分为五类&#xff0c;A,B,C,D,E,…

我的数据仓库与数据挖掘期末大作业重置版

文章目录 我的数据仓库与数据挖掘期末大作业重置版准备工作预设定及导入相对应的库库的导入调整 Jupyter Notebook 的预设定调整 MatPlotLib 和 Pandas 的输出设置 任务 1&#xff1a;预测问题数据的保存和读取数据的分析和预处理模型的选择和构建线性回归一元多项式回归 拟合预…

CUDA C编程权威指南习题解析

文章目录 一、1.6节习题二、2.6习题三、四、五、六、 一、1.6节习题 1.参考图1-5&#xff0c;分析以下几种数据划分形式&#xff1a; &#xff08;1&#xff09;对于二维数据&#xff0c;沿x轴进行块划分 &#xff08;2&#xff09;对于二维数据&#xff0c;沿y轴进行周期划…

cocos creator 的 widget组件的使用及踩坑

以下的内容基于cocos creator 3.8版本&#xff0c;如有错误&#xff0c;恳请指出。 &#x1f449;官方文档的指引 应用&#xff1a;以上官方指引有非常清晰的使用方式&#xff0c;接下来说明一些注意事项&#xff1a; 1、与canvas搭配的使用&#xff0c;解决多分别率适配问题。…

PHP搭建环境

一、安装apache 1、获取Apache安装软件 2、双击安装即可:指定对应的路径:E:server/apache 3、选择安装模式:使用自定义模式 4、选择安装位置 二、Apache的目录结构说明 三、Httpd.exe的详细应用 1、服务器进程:运行之后才能够工作

微积分复习笔记 Calculus Volume 2 - 4.1 Basics of Differential Equations

4.1 Basics of Differential Equations - Calculus Volume 2 | OpenStax

0003.基于springboot的“共享书角”图书借还管理系统

适合初学同学练手项目&#xff0c;部署简单&#xff0c;代码简洁清晰&#xff1b; 一、系统架构 前端&#xff1a;vue| elementui| 微信小程序 后端&#xff1a;springboot | mybatis-plus 环境&#xff1a;jdk1.8 | mysql | maven 系统设计说明: 二、代码及数据库 1.管理…

python学opencv|读取图像(十二)BGR图像转HSV图像

【1】引言 前述已经学习了opencv中图像BGR相关知识&#xff0c;文章链接包括且不限于下述&#xff1a; python学opencv|读取图像&#xff08;六&#xff09;读取图像像素RGB值_opencv读取灰度图-CSDN博客 python学opencv|读取图像&#xff08;七&#xff09;抓取像素数据顺利…

音频进阶学习八——傅里叶变换的介绍

文章目录 前言一、傅里叶变换1.傅里叶变换的发展2.常见的傅里叶变换3.频域 二、欧拉公式1.实数、虚数、复数2.对虚数和复数的理解3.复平面4.复数和三角函数5.复数的运算6.欧拉公式 三、积分运算1.定积分2.不定积分3.基本的积分公式4.积分规则线性替换法分部积分法 5.定积分计算…

智能高效的IDE GoLand v2024.3全新发布——支持最新Go语言

GoLand 使 Go 代码的阅读、编写和更改变得非常容易。即时错误检测和修复建议&#xff0c;通过一步撤消快速安全重构&#xff0c;智能代码完成&#xff0c;死代码检测和文档提示帮助所有 Go 开发人员&#xff0c;从新手到经验丰富的专业人士&#xff0c;创建快速、高效、和可靠的…

SQL server学习06-查询数据表中的数据(中)

目录 一&#xff0c;聚合函数 1&#xff0c;常用聚合函数 2&#xff0c;具体使用 二&#xff0c;GROP BY子句分组 1&#xff0c;基础语法 2&#xff0c;具体使用 3&#xff0c;加上HAVING对组进行筛选 4&#xff0c;使WHERE记录查询条件 汇总查询&#xff1a;在对数…

游戏引擎学习第52天

仓库 : https://gitee.com/mrxiao_com/2d_game 这节的内容相当多 回顾 在游戏中&#xff0c;实体被分为不同的类别&#xff1a;接近玩家的“高频实体”、距离较远并正在模拟的“低频实体”和不进行更新的“休眠实体”。这些实体会根据它们与玩家的距离进行处理&#xff0c;接…

websocket_asyncio

WebSocket 和 asyncio 指南 简介 本指南涵盖了使用 Python 中的 websockets 库进行 WebSocket 编程的基础知识&#xff0c;以及 asyncio 在异步非阻塞 I/O 中的作用。它提供了构建高效 WebSocket 服务端和客户端的知识&#xff0c;以及 asyncio 的特性和优势。 1. 什么是 WebS…

ARCGIS国土超级工具集1.2更新说明

ARCGIS国土超级工具集V1.2版本&#xff0c;功能已增加至47 个。在V1.1的基础上修复了若干使用时发现的BUG&#xff0c;新增了"矢量分割工具"菜单&#xff0c;同时增加及更新了了若干功能&#xff0c;新工具使用说明如下&#xff1a; 一、勘测定界工具栏更新界址点成果…