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

【Tomcat】第五站:Servlet容器

        Tomcat启动后,获取到项目当中所有的servlet的@WebServlet中的配置信息。将配置信息和类对象都写入一个map集合当中。

        map就是一个key-value类型的集合。

MyTomcat中我们获取到了类对象和注解值。

Tomcat与请求连通

1. ServletConfigMapping

        1. 创建一个config包,包下新建一个ServletConfigMapping类,写一个static代码块,static代码块在main方法执行之前执行。把MyTomcat中的代码搬过来。

package com.qcby.tomcat.config;

import com.qcby.tomcat.webServlet.WebServlet;

import java.io.File;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;

/*
* servlet容器
*
*
* */
public class ServletConfigMapping {

    //static代码块在main方法执行之前执行
    static{
        try {
            // 1. 扫描包路径 ()
            String packageName = "com.qcby.tomcat.myweb";
            //通过调用getclass方法,获取到了myweb这个包下的所有类的类对象,并将其放入到了容器当中
            List<Class<?>> classes = getClasses(packageName);

            // 2. 遍历所有类,检查是否有@WebServlet注解
            for (Class<?> clazz : classes) {
                if (clazz.isAnnotationPresent(WebServlet.class)) {//检查是否含有@WebServlet注解
                    // 3. 获取@WebServlet注解的值
                    WebServlet webServlet = clazz.getAnnotation(WebServlet.class);
                    System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.path());
                }
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
    /**
     * 获取指定包下的所有类
     *
     * @param packageName 包名,例如 "com.qcby.tomcat.myweb"
     * @return 类对象列表
     * @throws Exception
     */
    private static List<Class<?>> getClasses(String packageName) throws Exception {
        List<Class<?>> classes = new ArrayList<>();//定义一个可变长数组,放置类对象

        String path = packageName.replace('.', '/'); // 将包名转换为文件路径

        // 通过类加载器获取包的资源路径 ClassLoader是类加载器,负责动态加载Java类到Java虚拟机(JVM)中。
        ClassLoader classLoader = Thread.currentThread().getContextClassLoader();
        //Thread.currentThread() 方法返回对当前执行线程的引用。
        //.getContextClassLoader() 方法则返回该线程的上下文类加载器。上下文类加载器是线的程在创建时从父线程继承,
        //                              或者在创建线程时没有设置父线程的上下文类加载器时,默认使用应用程序类加载器(Application ClassLoader)

        //ClassLoader 类提供了 getResources(String name) 方法,该方法用于---查找具有指定名称的资源
        Enumeration<URL> resources = classLoader.getResources(path);
        //Enumeration<URL> 对象包含了所有找到的资源的URL。


        while (resources.hasMoreElements()) {//通过hasMoreElements()方法来检查是否还有更多的元素
            如果返回true,则表示枚举中还有元素未被遍历,可以继续调用nextElement()方法来获取下一个元素
            URL resource = resources.nextElement();

            //调用URL对象的toURI()方法将其转换为URI对象。
            //使用new File(URI uri)构造函数将URI对象转换为File对象。
            File directory = new File(resource.toURI());

            //File对象代表目录,不是文件
            // 扫描文件夹下的所有类文件
            if (directory.exists()) {
                for (File file : directory.listFiles()) {//可以利用listFiles()方法来获取该目录下所有文件和子目录的数组
                    if (file.getName().endsWith(".class")) {
                        //java文件自己创建的话是xx.java文件,后经过编译是:xx.class文件,这里用的是编译后的文件
                        // 获取类的完整类名
                        String className = packageName + "." + file.getName().replace(".class", "");
                        classes.add(Class.forName(className));
                    }
                }
            }
        }
        return classes;
    }

    public static void main(String[] args) {

    }
}

        执行这个main方法,就自动会先执行static代码块。通常是这样用的。(也可以直接写在main方法里)。

        2. 创建map映射表,类对象用HttpServlet类型承接。

!!为什么这里是HttpServlet?

for (Class<?> clazz : classes) {
    if (clazz.isAnnotationPresent(WebServlet.class)) {//检查是否含有@WebServlet注解
        // 3. 获取@WebServlet注解的值
        WebServlet webServlet = clazz.getAnnotation(WebServlet.class);
        //System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.path());
        
        //classMap.put(webServlet.path(),(Class<HttpServlet>) clazz);
    }
}

这里的类对象:

        MyFirstServlet的类对象

        MySecondServlet的类对象

        MyThirdServlet的类对象

        这几个都继承了HttpServlet,都是HttpServlet的子类。

        我们就可以用父类的容器去承接子类的对象,这样的话就可以承接这几个,如果写MyFirstServlet,就不能写其他的类,无法转型。父类的引用指向子类的对象,子类可以向上转型为父类的引用。他们之间没有父子类的关系。

        故写HttpServlet。

 代码修改:

public static Map<String,Class<HttpServlet>> classMap=new HashMap<>();
//static代码块在main方法执行之前执行
static{
    try {
        // 1. 扫描包路径 ()
        String packageName = "com.qcby.tomcat.myweb";
        //通过调用getclass方法,获取到了myweb这个包下的所有类的类对象,并将其放入到了容器当中
        List<Class<?>> classes = getClasses(packageName);

        // 2. 遍历所有类,检查是否有@WebServlet注解
        for (Class<?> clazz : classes) {
            if (clazz.isAnnotationPresent(WebServlet.class)) {//检查是否含有@WebServlet注解
                // 3. 获取@WebServlet注解的值
                WebServlet webServlet = clazz.getAnnotation(WebServlet.class);
                //System.out.println("类名: " + clazz.getName() + " | URL路径: " + webServlet.path());

                classMap.put(webServlet.path(),(Class<HttpServlet>) clazz);
            }
        }

 2. MyTomcat

        当执行MyTomcat的main方法时,ServletConfigMapping并没有执行。

        启动和后两个没有连接上。

        我们想要当启动Tomcat时,就能获取到项目当中所有的配置信息,根据生成类对象,并且把类对象放入到容器中。

        即,想要当MyTomcat的main方法执行时,ServletConfigMapping也执行。

        方法很简单,只需要在ServletConfigMapping写一个静态方法,什么名字都可以,在MyTomcat调用即可。

        因为static代码块在main方法之前执行,所以在static代码块执行之后,定义的www方法(改名init())才会执行。

        或者直接把static代码块中的代码放进init方法中。

3. Server

        同样,在Server类中,把main方法改成serverInit方法(把该方法直接调到MyTomcat类中)

public static void serverInit() throws Exception{
    // 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信
    ServerSocket serverSocket = new ServerSocket(8080);//监听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();
    }
}

然后,在MyTomcat中调用。

public class MyTomcat {
    public static void main(String[] args) {
        try{
            ServletConfigMapping.init();//初始化servlet容器
            Server.serverInit();//启动server服务
        }catch (Exception e){
            e.printStackTrace();
        }

    }

         现在,请求也打过来了,该执行下一步,根据key值,获取类对象。

4. 如何做好匹配?

        将请求的信息封装到了request对象当中,根据request当中的path,去解决掉当前匹配问题。

        当程序启动之后,已经把map集合创建好了,此时用户的请求打过来了,请求信息处理完之后封装在request对象当中,

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 com.qcby.tomcat.webServlet.WebServlet;

import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.lang.annotation.Annotation;
import java.net.ServerSocket;
import java.net.Socket;
import java.net.URL;
import java.util.ArrayList;
import java.util.Enumeration;
import java.util.List;

public class MyTomcat {
    public 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 serverInit() throws Exception{
        // 1.打开通信端口   tomcat:8080   3306  ---------》进行网络通信
        ServerSocket serverSocket = new ServerSocket(8080);//监听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, InstantiationException, IllegalAccessException {
        //将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 path=firstLine.split("\\s")[1];
            String method=firstLine.split("\\s")[0];
            System.out.println(path+" "+method);
            request.setMethod(method);
            request.setPath(path);
        }

        dis(request);
    }

    public static void dis(Request request) throws InstantiationException, IllegalAccessException {
        if(!request.getPath().equals("")){//不是空请求
            if(ServletConfigMapping.classMap.get(request.getPath())!=null){//当前请求地址能否从classMap中查找到
                Class<HttpServlet> ClassServlet=ServletConfigMapping.classMap.get(request.getPath());//获取类对象
                HttpServlet servlet=ClassServlet.newInstance();//根据获取到的类对象,创建对应的对象  用父类去接,多态(父类的引用指向子类的对象)
                servlet.service(request,response);//调用HttpServlet的service方法,判断是get请求还是post请求
            }
        }
    }

}

其中,

public static void dis(Request request) throws InstantiationException, IllegalAccessException {
    if(!request.getPath().equals("")){//不是空请求
        if(ServletConfigMapping.classMap.get(request.getPath())!=null){//当前请求地址能否从classMap中查找到,匹配成功
            Class<HttpServlet> ClassServlet=ServletConfigMapping.classMap.get(request.getPath());//获取类对象
            HttpServlet servlet=ClassServlet.newInstance();//根据获取到的类对象,创建对应的对象  用父类去接,多态(父类的引用指向子类的对象)
            servlet.service(request,response);//调用HttpServlet的service方法,判断是get请求还是post请求
        }
    }
}

        方法是对象当中的一块内存空间

        但是这是类对象,不是对象,想要得到信息就得生成对象。

        通过类对象去创建对象,是我们需要做的。

        用类对象创建对象, 因为不确定是请求的哪个对象,需要用父类去承接(多态), 根据对象调用它的doGet或doPost方法。

如此,

        即可在启动Tomcat之后,通过浏览器输入请求,得到信息。

 


http://www.kler.cn/a/445860.html

相关文章:

  • 堆【Lecode_HOT100】
  • 投标心态:如何在“标海战术”中保持清醒的头脑?
  • APM32F411使用IIS外设驱动es8388实现自录自播
  • K8s 节点 NotReady 后 Pod的变化
  • Android Java Ubuntu系统如何编译出 libopencv_java4.so
  • OpenHarmony-4.HDI 框架
  • mfc140.dll是什么东西?mfc140.dll缺失的几种具体解决方法
  • 腾讯云云开发 Copilot 深度探索与实战分享
  • STM32单片机芯片与内部33 ADC 单通道连续DMA
  • 子域提取工具,子域名收集神器,支持多种数据源和枚举选项,域名发现工具,可以为任何目标枚举海量的有效子域名,安全侦察工具,利用证书透明原则监控部署的新子域
  • html在线转换工具集合大全
  • AFL-Fuzz 的使用
  • 五十个网络安全学习项目——(九)无线网络安全分析
  • windows 钉钉缓存路径不能修改 默认C盘解决方案
  • Python 【大模型】之 使用千问Qwen2-VL 大模型训练LaTeX数学公式图,并进行LaTeX图识别测试
  • 校园快领系统|Java|SSM|VUE| 前后端分离
  • 模型训练之优化器
  • #渗透测试#漏洞挖掘#红蓝攻防#护网#sql注入介绍05-基于堆叠查询的SQL注入(Stacked Queries SQL Injection)
  • java全栈day17--Web后端实战(java操作数据库)
  • springboot3访问第三方接口
  • GNU Octave:特性、使用案例、工具箱、环境与界面
  • PHP接入美团联盟推广
  • textfile类型小文件合并
  • Unity动态读取外部图片转Texture2D,内存过大问题解决方案
  • [ThinkPHP]5.0.23-Rce 1
  • Oracle/MySQL 到 OceanBase 数据库迁移的关键问题与解决方案