【学习笔记】手写 Tomcat 七

news2025/1/4 17:13:17

目录

一、优化 Dao 

1. 设置 UserDaoImpl 为单例模式

2. 创建 Dao 工厂

3. 在 Service 层获取 UserDao 的实例

二、优化 Service

1. 设置 UserServiceImpl 为单例模式

2. 创建 Service 工厂

3. 在 Servlet 层获取 Service 实现类的对象

三、优化 Servlet

1. 使用配置文件

1. 创建配置文件

2. 创建工具类

3. 调用工具类获取对象

测试

2. 使用注解的方式

1. 创建自定义注解

2. 使用注解

3. 创建工具类

4. 获取对象

测试

四、作业

1. 绘制架构图

2. 了解 NIO


一、优化 Dao 

1. 现在在 Service 层,调用Dao层的实现类都要 new 一下,这样就会创建很多对象,比较占内存,比如 Service 层的其他实现类也会调用 UserDao,那么只需要创建一次 UserDao 的对象就行了

使用单例模式

2. 既然 Service 层的很多方法可能也会调用UserDao,如果以后需要更换 UserDao的实现类,那么所有调用 UserDao 的地方都需要修改一下,这样比较麻烦

使用工厂模式获取 UserDao 的对象,如果要修改,只需要在工厂类修改即可

1. 设置 UserDaoImpl 为单例模式

2. 创建 Dao 工厂

用于获取 Dao 的不同实例

3. 在 Service 层获取 UserDao 的实例

二、优化 Service

和 Dao 层同理,在 Servlet 层也多次创建了 UserService 的实现类,比如登录和修改密码都用到了UserService

1. 设置 UserServiceImpl 为单例模式

2. 创建 Service 工厂

package com.shao.Service;

import com.shao.Service.Impl.UserServiceImpl;

public class ServiceFactory {
    
    public static UserService getUserService() {
        return UserServiceImpl.getInstance();
    }
}

3. 在 Servlet 层获取 Service 实现类的对象

三、优化 Servlet

现在的情况是当 Tomcat 接收一个请求后,会先判断是请求的静态资源还是动态资源,如果是动态资源,还要继续判断请求的是哪个功能,然后调用相应的 Servlet 执行

这样有些缺点,当功能很多时,一个个判断请求的是哪个功能,效率不高,而且代码不够优雅,并且来一个请求就会创建一下 Servlet 对象,比较消耗资源,那如何解决呢?

解决方案是:我们先把所有的 Servlet 对象创建好,然后放到一个容器(集合)里,当有请求时,取出对应的 Servlet 对象执行。这是不是很熟悉的感觉?池化思想

那么问题来了,系统怎么知道哪些 Servlet 需要创建对象?

1. 使用配置文件

把要创建的 Servlet 的全类名放到配置文件,然后读取配置文件,通过反射技术创建对象

1. 创建配置文件

2. 创建工具类

package com.shao.Utils;

import com.shao.Servlet.BaseServlet;

import java.io.File;
import java.io.FileInputStream;
import java.util.HashMap;
import java.util.Properties;
import java.util.Set;

public class ServletUtil {

    // 获取当前系统分隔符
    final static String FENGEFU = File.separator;
    // 存放 Servlet 对象
    static HashMap<String, BaseServlet> map = new HashMap<>();
    // 读取配置文件的内容
    static Properties properties = new Properties();

    static {
        try {
            properties.load(new FileInputStream("config" + FENGEFU + "Servlet.properties"));

            // 把 Key 放到集合中
            Set<Object> keySet = properties.keySet();
            for (Object key : keySet) {
                String value = properties.getProperty(key.toString());

                // 通过全类名获取 Class 对象,Class 对象记录了这个类的全部信息
                Class<?> aClass = Class.forName(value);

                /*
                 *   底层原理:
                 *   1. 调用构造器,通过 Class 对象找到对应类型的无参构造器
                 *   2. 实例化对象:使用构造器创建一个新的对象实例
                 *   3. 返回实例
                 * */
                BaseServlet baseServlet = (BaseServlet) aClass.newInstance();

                // 添加到 map 集合
                map.put(key.toString(), baseServlet);

            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    // 对外提供一个接口,获取集合中 key 对应的 value ,value 是 Servlet 对象
    public static BaseServlet getServletClass(String key) {
        return map.get(key);
    }

}

3. 调用工具类获取对象

在响应类调用 ServletUtil 获取Servlet 实例

package com.shao.net;

import com.alibaba.fastjson2.JSON;
import com.shao.Servlet.BaseServlet;
import com.shao.Servlet.ChangePasswordServlet;
import com.shao.Servlet.LoginServlet;
import com.shao.Utils.DBConnectUtil;
import com.shao.Utils.ServletUtil;
import com.shao.Utils.responseDTO;

import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.sql.*;

public class HttpResponse {

    /**
     * 输出流
     */
    private OutputStream os;

    /**
     * 解析请求信息的对象
     */
    private HttpRequest httpRequest;

    public HttpResponse(OutputStream os, HttpRequest httpRequest) {
        this.os = os;
        this.httpRequest = httpRequest;
    }

    public void response(String filePath) {
        //判断请求的是否为静态文件
        if (StaticResourceHandler.isLikelyStaticResource(httpRequest.getRequestModule())) {
            // 获取静态资源一般是 GET 请求方法
            if (httpRequest.getRequestMethod().equals("GET")) {
                // 响应静态资源
                responseStaticResource(filePath);
            }
        } else {

            // 处理动态请求
            System.out.println("请求动态资源");

            // 获取 Servlet 对象,参数是请求的模块名
            BaseServlet servlet = ServletUtil.getServletClass(httpRequest.getRequestModule());

            // 如果没有找到对应的 Servlet ,返回 404
            if (servlet == null) {
                responseStaticResource("webs/pages/not_Found404.html");
                return;
            }
            // 调用 service 方法
            servlet.service(httpRequest, this);

//            if (httpRequest.getRequestModule().equals("/doLogin")) {
//
//                // 创建 登录的 Servlet 对象
//                LoginServlet loginServlet = new LoginServlet();
//
//                // 调用 service 方法,响应数据也封装在 servlet 里
//                loginServlet.service(httpRequest, this);
//
//
//            } else if ("/ChangePassword".equals(httpRequest.getRequestModule())) {
//
//                ChangePasswordServlet servlet = new ChangePasswordServlet();
//
//                servlet.service(httpRequest, this);
//            } else if ("/test".equals(httpRequest.getRequestModule())) {
//                send(JSON.toJSONBytes(new responseDTO(200, "ok", "test")));
//            }
        }
    }

    /**
     * 响应静态资源
     */
    private void responseStaticResource(String filePath) {
        // 读取文件
        byte[] fileContents = StaticResourceHandler.getFileContents(filePath);

        // 判断文件是否存在,不存在则返回 404 的页面
        if (fileContents == null) {
            try {
                FileInputStream fis = new FileInputStream("webs/pages/not_Found404.html");

                fileContents = new byte[fis.available()];

                fis.read(fileContents);

                fis.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }

        // 响应协议
        String protocol = httpRequest.getRequestProtocol();
        // 文件媒体类型
        String fileMimeType = StaticResourceHandler.getFileMimeType(filePath);
        try {
            os.write((protocol + " 200 OK\r\n").getBytes());
            os.write(("Content-Type: " + fileMimeType + "\r\n").getBytes());
            os.write(("Content-Length: " + fileContents.length + "\r\n").getBytes());
            os.write("\r\n".getBytes());
            os.write(fileContents);
            os.flush();
            System.out.println("响应成功");
            os.close();

        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public void send(byte[] content) {
        // 获取请求协议
        String protocol = httpRequest.getRequestProtocol();
        try {
            os.write((protocol + " 200 OK\r\n").getBytes());
            os.write(("Content-Type: " + "text/html;charset=utf-8" + "\r\n").getBytes());
            os.write(("Content-Length: " + content.length + "\r\n").getBytes());
            os.write("\r\n".getBytes());
            os.write(content);
            os.flush();
            System.out.println("响应成功");
            os.close();

        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

测试

2. 使用注解的方式

在需要创建的 Servlet 的类名上加上自定义注解,然后通过反射技术创建对象

1. 创建自定义注解

package com.shao.Annotation;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface MyServlet {
    String value();
}

2. 使用注解

3. 创建工具类

package com.shao.Utils;

import com.shao.Annotation.MyServlet;
import com.shao.Servlet.BaseServlet;

import java.io.File;
import java.net.URISyntaxException;
import java.net.URL;
import java.util.HashMap;

public class ServletByAnnoUtil {
    // 存放 Servlet 对象
    private static HashMap<String, BaseServlet> map = new HashMap<>();
    // 要扫描包的路径
    private static String PackagePath = "com.shao.Servlet";

    static {
        // 获取类加载器
        ClassLoader classLoader = ServletByAnnoUtil.class.getClassLoader();

        // 获取包的完整路径
        URL resource = classLoader.getResource(PackagePath.replace(".", "/"));
        if (resource == null) {
            throw new RuntimeException("Package path not found:" + PackagePath);
        }
        try {
            // 创建目录对象
            File packageDir = new File(resource.toURI());

            // 获取目录下所有文件
            File[] files = packageDir.listFiles();
            /*
             *   1. 判断文件是否为 .class 文件
             *   2. 获取文件名,去掉 .class 后缀
             *   3. 包路径 + 文件名,拼接成全类名
             *   4. 获取 Class 对象
             *   5. 判断 Class 对象是否有 MyServlet 注解
             *   6. 如果有,判断是否 BaseServlet 类,或者继承 BaseServlet
             *   7. 创建对象,获取注解的值,存放到 map 中
             *
             * */
            for (File file : files) {
                if (file.getName().endsWith(".class")) {
                    // 获取全类名
                    String className = PackagePath + "." + file.getName().substring(0, file.getName().lastIndexOf("."));
                    // 获取 Class 对象
                    Class<?> aClass = Class.forName(className);
                    // 判断是否有 MyServlet 注解
                    if (aClass.isAnnotationPresent(MyServlet.class)) {
                        // 判断是否继承 BaseServlet
                        if (BaseServlet.class.isAssignableFrom(aClass)) {
                            // 创建对象
                            BaseServlet servlet = (BaseServlet) aClass.newInstance();
                            // 获取注解的值
                            MyServlet annotation = aClass.getAnnotation(MyServlet.class);
                            map.put(annotation.value(), servlet);
                        }
                    }
                }
            }
        } catch (URISyntaxException | ClassNotFoundException | InstantiationException | IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }

    // 对外提供一个接口,获取 Servlet 对象
    public static BaseServlet getServletClass(String key) {
        return map.get(key);
    }
}

4. 获取对象

测试

四、作业

1. 绘制架构图

2. 了解 NIO

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

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

相关文章

Map+Set

我们前面接触过的string、vector、list这些都算序列式容器&#xff0c;它们都有一定的关联性&#xff0c;即使随便换位置也无伤大雅&#xff0c;因为是它们靠位置顺序来保存的。但是今天的MapSet就不是了&#xff0c;它们算关联式容器&#xff0c;两个位置之间有紧密的联系&…

Linux-df命令使用方法

Linux-df&#xff08;disk filesystem&#xff09;命令 df 命令是 Unix 和 Linux 系统中用于报告文件系统磁盘空间使用情况的工具。 df [OPTION]... [FILE]...OPTION 常用选项&#xff08;博主一般df -h用的较多&#xff0c;可读性较好&#xff09; -h&#xff1a;以人类可读的…

Uniapp 微信小程序 最新 获取用户头像 和 昵称 方法 有效可用

文章目录 前言代码实现运行效果技术分析 前言 同事有个需求 授权获取用户头像 和 昵称 。之前做过线上小程序发版上线流程 就实现了下 最新的方法和 api 有些变化 记录下 代码实现 先直接上代码 <template><view class"container"><buttonclass&qu…

如何部署北斗定位应用,基于国产自主架构LS2K1000LA-i处理器平台

北斗卫星导航系统(以下简称北斗系统)是着眼于国内经济社会发展需要,自主建设、独立运行的卫星导航系统。经过多年发展,北斗系统已成为面向全球用户提供全天候、全天时、高精度定位、导航与授时服务的重要新型基础设施。 图 1 北斗定位系统的应用优势 强可控:北斗系统是国…

Ollama本地部署大模型及应用

文章目录 前言一、下载安装1.Mac2.Windows3.linux4.docker5.修改配置&#xff08;可选&#xff09;1.linux系统2.window 系统3.mac系统 二、Ollama使用1.命令2.模型下载3.自定义模型4.API 服务 三、Open WebUI 使用四、Dify使用 前言 Ollama 是一个专注于本地部署大型语言模型…

第四届摩纳哥智能化可持续发展游艇码头交流会

第四届摩纳哥智能化可持续发展游艇码头交流会 游艇生态和经济转型 2024年9月23日&#xff0c;第四届摩纳哥智能化可持续发展游艇码头交流会于摩纳哥游艇俱乐部顺利落幕。该交流会由摩纳哥游艇码头顾问公司&#xff08;M3&#xff09;主办&#xff0c;吸引了全球250名游艇行业领…

数据集-目标检测系列-口罩检测数据集 mask>> DataBall

数据集-目标检测系列-口罩检测数据集 mask>> DataBall 数据集-目标检测系列-口罩检测数据集 mask 数据量&#xff1a;1W DataBall 助力快速掌握数据集的信息和使用方式&#xff0c;享有百种数据集&#xff0c;持续增加中。 数据项目地址&#xff1a; gitcode: https…

JAVA笔记 | 实际上用到的策略模式(可直接套用)

自己开发中用到了策略模式&#xff0c;这样写不一定是最好的&#xff0c;但是满足了业务场景跟使用要求&#xff0c;做个笔记&#xff0c;下次有用到可以快速复习跟套用 假设使用场景&#xff1a;有几只宠物&#xff0c;猫跟狗等&#xff0c;要求他们做各种动作&#xff0c;比如…

无人机之物流货运篇

一、无人机货运的崛起 随着电商、快递行业的蓬勃发展&#xff0c;传统的地面物流已经难以满足日益增长的快递量和对速度的追求。而无人机货运凭借其高效、快捷、灵活的特点&#xff0c;逐渐成为了物流行业的新宠。无人机可以在城市上空快速穿梭&#xff0c;不受地面交通拥堵的限…

语言模型发展史

四个阶段 第一阶段&#xff1a;基于规则和统计的语言模型 由人工设计特征并使用统计方法对固定长度的文本窗口序列进行建模分析&#xff0c;这种建模方式也被称为N-gram语言模型。 优点&#xff1a; 1&#xff09;采用极大似然估计, 参数易训练 2&#xff09;完全包含了前n-…

Arthas vmtool(从 jvm 里查询对象,执行 forceGc)

文章目录 二、命令列表2.1 jvm相关命令2.1.15 vmtool&#xff08;从 jvm 里查询对象&#xff0c;执行 forceGc&#xff09;举例1&#xff1a;获取对象 二、命令列表 2.1 jvm相关命令 2.1.15 vmtool&#xff08;从 jvm 里查询对象&#xff0c;执行 forceGc&#xff09; vmtoo…

nodejs 014: React.FC 与 Evergreen(常青树) React UI 框架的的Dialog组件

React.FC React.FC是React中用于定义函数组件“Function Component”的类型。它代表&#xff0c;可以帮助你在TypeScript中提供类型检查和自动补全。使用React.FC时&#xff0c;可以明确指定组件的props类型&#xff0c;并且它会自动推导children属性。下面是一个使用 React.F…

微服务-- Gateway服务网关

Spring Cloud Gateway 是 Spring Cloud 的一个全新项目&#xff0c;该项目是基于 Spring 5.0&#xff0c;Spring Boot 2.0 和 Project Reactor 等响应式编程和事件流技术开发的网关&#xff0c;它旨在为微服务架构提供一种简单有效的统一的 API 路由管理方式。 为什么需要网关…

从0开始linux(5)——vim

欢迎来到博主的专栏&#xff1a;从0开始linux 博主ID&#xff1a;代码小豪 文章目录 vim的多种模式底行模式命令命令模式视块模式&#xff08;visual block&#xff09; vim的配置 vim是linux系统的文本编辑器。就像windows的记事本一样。 使用vim指令可以使用vim打开一个文本文…

MacOS多桌面调度快捷键

单桌面调度快捷键 可能是我用着妙控鼠标用着不习惯&#xff0c;所以追求快捷键操作&#xff0c;看起来也比较酷。而且在Windows上&#xff0c;我基本不使用多桌面&#xff0c;但是看着同事用Mac的多桌面用的飞起&#xff0c;炫酷程度不亚于win7的Windows键Tab。在不使用多桌面的…

小川科技携手阿里云数据库MongoDB:数据赋能企业构建年轻娱乐生态

随着信息技术的飞速发展&#xff0c;企业在处理海量数据时所面临的挑战日益严峻。特别是在年轻娱乐领域&#xff0c;用户行为的多样性和数据量的激增对数据存储与分析技术提出了更高的要求。在此背景下&#xff0c;小川凭借其前瞻性的技术视野&#xff0c;选择了MongoDB作为其数…

乒乓buffer(国科微笔试填空)_2024年9月26日

乒乓buffer是否提高了并行度&#xff1f;是 流水线式处理&#xff0c;提高并行度 位宽为4&#xff0c;深度为8的sram&#xff1a; 当sel拉高时&#xff0c;a、b模块在wren和rden有效时分别写和读&#xff1b;当sel拉低时&#xff0c;a、b模块在rdenwren有效时分别读和写

远程访问软路由

远程访问软路由主要涉及通过互联网从远程位置访问和控制基于软件的路由器系统。以下是远程访问软路由的一般方法&#xff1a; 一、远程访问软路由的方法 通过Web管理界面访问&#xff1a; 适用于大多数支持Web管理的软路由系统。用户只需在浏览器中输入软路由的公网IP地址或域…

【C语言】手把手带你拿捏指针(完)(指针笔试、面试题解析)

文章目录 一、sizeof和strlen的对⽐1.sizeof2.strlen3.sizeof与strlen对比 二、数组和指针笔试解析1.一维数组2.字符、字符串数组和字符指针代码1代码2代码3代码4代码5代码6 3.二维数组4.总结 三、指针运算笔试题解析代码1代码2代码3代码4代码5代码6 一、sizeof和strlen的对⽐ …

freeRDP OPenssl

libusb需要下载 我使用的是VS2019编译 所以需要include 与vs2019 在cmake里面修改路径 C:/Users/JPM/source/repos/freeRDP/FreeRDP-stable-2.0/libusb-1.0.24/include/libusb-1.0 C:/Users/JPM/source/repos/freeRDP/FreeRDP-stable-2.0/libusb-1.0.24/VS2019/MS64/static/l…