从零开始写一个Tomcat(贰)--建立动态服务器

上文书说道如何通过http协议建立一个静态的服务器来访问静态网页,但我们选择tomcat最主要的原因还是因为它能动态的执行servlet,这边文章将引导你实现一个能够运行servlet的服务器,这个简易的服务器符合tomcat的基本原理,但真的tomcat远不是这么简单,即使是15年前的tomcat4。

1.主程序逻辑分离

既然要实现动态的服务器,首先我们要实现模式的识别,在tomcat中,tomcat通过读取web.xml,在servlet的配置中通过url的模式来匹配servlet,我们这里当然不会这么复杂,我们暂且设置如下规则,如果你想访问一个servlet,那么你就在浏览器输入如localhost/servlet/TestServlet这样的地址,TestServlet是你servlet的名字,准确点来说就是你的servlet编译后的class的名称,当然你可以起你任意喜欢的名字

以上文为基础,将main方法中的静态处理改成如下

 if(request.getUri().startsWith("/servlet/"))
{
       ServletProcessor processor=new ServletProcessor();
        processor.process(request,response);
}else{
        StaticResponseProcessor processor=new StaticResponseProcessor();
         processor.process(request,response);
}

当request获得的uri以servlet开头,就去访问ServletProccess的方法.

2.ServletProccess

这个类的职责就是用于找到servlet并调用.不多说,先上代码

    public class ServletProcessor {
    public void process(Request request,Response response)
    {
        String uri=request.getUri();
        String servletName=uri.substring(uri.lastIndexOf("/")+1);
        URLClassLoader loader=null;
        try
        {
            URL[] urls=new URL[1];
            URLStreamHandler streamHandler=null;
            File classpath=new File(Constants.WEB_ROOT);
            String repository=(new URL("file",null,
                    classpath.getCanonicalPath()+File.separator).toString());
            System.out.println(repository);
            urls[0] = new URL(null,repository,streamHandler);
            loader=new URLClassLoader(urls);
        } catch (MalformedURLException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        Class myClass=null;
        try{
            myClass=loader.loadClass(servletName);
        }catch (ClassNotFoundException e){
            e.printStackTrace();
        }
        Servlet servlet=null;

        try{
            servlet=(Servlet)myClass.newInstance();
            servlet.service(request,response);

        }catch (Exception e){
            System.out.println(e.toString());
        }catch (Throwable t){
            System.out.println(t.toString());
        }

    }
}

首先,我们要找到这个类,这个类应该在哪呢,在tomcat中,这个类即class文件在/WEB-INF/classes下,我们暂且先直接放在webroot下面,负责记录存储位置的是一个专门的Constants类的静态变量

 public static final String WEB_ROOT=System.getProperty("user.dir")+File.separator+"webroot";

  程序使用一个URLClassLoader来加载这个class,为了禁止servlet访问tomcat中的类,tomcat自定义了一个加载器,这个问题以后再详细说明,加载完类之后,用newInstance实例化,按理说这里会先调用init方法,老规矩,先忽略了,然后调用servlet的service方法,这时候细心的你可能发现了,在servlet的方法中,我的request是实现了servletRequest接口的,同理还有response,但你的request是自己定义的,这么调用肯定是错的,而且说好的response.getWriter呢,下面我们来改造request和response

3.request,response

其实并不需要多少操作,只需要把request实现ServletRequest接口,response实现ServletResponse接口,然后使用IDE的代码补全,里面的方法先都空着好了,在response的getWriter添加如下代码

 public PrintWriter getWriter() throws IOException {
        writer=new PrintWriter(output,true);
        return writer;
    }

4.facade

至此,这个动态服务器已经可以用了,但是不知道你有没有注意到另一个问题,我们在servlet使用的request是tomcat传给我们的,那也就是说我们可以把这个servletRequest向下转型为tomcat的request类,然后我们就可以访问request的parse()之类的方法,但显然tomcat是不容许的,tomcat在这里使用了外观模式新建一个类requestFacade,在构造函数中传入tomcat的request,实现所有外部能够访问的方法,在方法中调用request的相应的方法,这样就能完美的隐藏那些request的必须的且不应该被servlet访问的public方法.

5.结果

public void service(ServletRequest req, ServletResponse res) throws ServletException, IOException {    System.out.println("service");    res.getWriter().println("service");}


地址:https://github.com/Asens/AsServer/tree/master/AsServerV1.0.1

下集预告:我会把接受请求和处理请求的逻辑分离,并引入线程池概念,是这个服务器能够同时处理多个请求
时间: 2024-12-20 00:07:25

从零开始写一个Tomcat(贰)--建立动态服务器的相关文章

从零开始写一个Tomcat(叁)--请求解析

挖坑挖了这么长时间也该继续填坑了,上文书讲到从零开始写一个Tomcat(贰)--建立动态服务器,讲了如何让服务器解析请求,分离servlet请求和静态资源请求,读取静态资源文件输出或是通过URLClassLoader找到我们请求的servlet,反射生成对应的实例,调用其service方法,传递初级解析的request和response,完成请求. 这很tomcat,but too simple 阅读本文,你将了解 连接器(connector),处理器(processor)逻辑分离 如何高效的解

深入浅出React Native 3: 从零开始写一个Hello World

这是深入浅出React Native的第三篇文章. 1. 环境配置 2. 我的第一个应用 将index.ios.js中的代码全部删掉,为什么要删掉呢?因为我们准备从零开始写一个应用~学习技术最好的方式就是自己动手写,看别人的代码一百遍的效果也不如自己写一遍来的效果大~ 我们要做的事情主要分成以下两步: 1. 创建组件 2. 将创建好的组件显示在app上 打开index.ios.js文件,输入 var HelloWorld = React.createClass({ render: functio

从零开始写一个arm下的裸板程序

从零开始写一个arm下的裸板程序.我们整个程序是基于uboot运行的. 所有我们可以借助uboot中的printf来输出,默认开发版的标准输出是串口. 电脑的默认标准输出的屏幕. 1.需要创建的文件由include文件夹,用来存放头文件. 2.创建一个hw.h头文件. 3.编写一个common.h,它定义了借用uboot的printf的宏.和NULL这个宏的定义. 4.hw.c 硬件相关的文件. 5.main.c c文件. 6.start.s 汇编文件. 7.ld.lds 链接脚本, 8.Mak

设计模式 - 动态代理原理及模仿JDK Proxy 写一个属于自己的动态代理

本篇文章代码内容较多,讲的可能会有些粗糙,大家可以选择性阅读. 本篇文章的目的是简单的分析动态代理的原理及模仿JDK Proxy手写一个动态代理以及对几种代理做一个总结. 对于代理模式的介绍和讲解,网上已经有很多优质的文章,我这里就不会再过多的介绍了,这里推荐几篇优质的文章作为参考: 给女朋友讲解什么是代理模式 轻松学,Java 中的代理模式及动态代理 另外,我的 github 仓库对应目录中也有相关的基础示例代码:https://github.com/eamonzzz/java-advance

怎么从零开始写一个秒杀项目

一,环境搭建 技术选型,该秒杀项目是采用springboot2.0和springCloud来开发的 1首先搭建父项目,所有的模块都是依赖该父项目 2搭建服务注册中心模块(Eureka) 3秒杀用户模块(集成mybatis,web,thymeleaf等),相当于买家 4公共模块,对应的是数据库表对应的实体类,form表单对应的pojo,工具类等 5商家模块,(集成mybatis,web,thymeleaf等),相当于卖家 6商铺模块,商家入驻后申请商铺 7商品类别模块,商品的层级分类 8商品服务模

c语言:写一个函数建立一个有3名学生数据的单向动态链表

写一个函数建立一个有3名学生数据的单向动态链表. 解:程序: #include<stdio.h> #include<stdlib.h> #define LEN sizeof(struct Student) struct Student { long num; float score; struct Student *next; }; int n; struct Student *creat(void)//定义函数返回一个指向链表头的指针 { struct Student *head

如何写一个Web服务器

最近两个月的业余时间在写一个私人项目,目的是在Linux下写一个高性能Web服务器,名字叫Zaver.主体框架和基本功能已完成,还有一些高级功能日后会逐渐增加,代码放在了github.Zaver的框架会在代码量尽量少的情况下接近工业水平,而不像一些教科书上的toy server为了教原理而舍弃了很多原本server应该有的东西.在本篇文章中,我将一步步地阐明Zaver的设计方案和开发过程中遇到的困难以及相应的解决方法. 为什么要重复造轮子 几乎每个人每天都要或多或少和Web服务器打交道,比较著名

从零开始写个一个豆瓣电影 (小程序教程1)

微信小程序内测至今也有20天左右,也有很多人作出了很多不错的DEMO并发布到github了.前几日看见了豆瓣电影这个demo,感觉很不错,也跟着做了一个,作为复习巩固文档API用. 废话不多说,直接进入正题: 第一节只写一个首页的展示,数据用的是自己写的虚拟的数据 新建一个demo,不要使用微信自带的DEMO,直接从零开始写起: 首先创建3个文件: app.json app.js apps.wxss app.json  : 主要写配置项:内容如下,具体的每个key值对应的意思,这里就不再细说了,

写一个简易web服务器、ASP.NET核心知识(4)

前言 昨天尝试了,基于对http协议的探究,我们用控制台写了一个简单的浏览器.尽管浏览器很low,但是对于http协议有个更好的理解. 说了上面这一段,诸位猜到我要干嘛了吗?(其实不用猜哈,标题里都有,又都不瞎...我就是调侃一下,说些没营养的笑话.我认为这样能不那么枯燥,尽管不好笑吧,但这不重要!) 没错,今天要尝试的东西,是自己写一个web服务器.初衷依旧和昨天一样,旨在理解一些东西,而不是真的写出一个多牛的东西. 第一次尝试(V1.0) 1.理论支持 其实关于http协议的理论方面我在<写