草庐IT

day02-实现01

liyuelian 2023-04-19 原文

实现01

1.实现任务阶段1

编写mytomcat,该服务器能给浏览器返回“你好,我是服务器!”的简单信息。

根据之前的tomcat框架整体分析,我们将浏览器发送请求,tomcat服务器处理请求,返回资源的整个过程分为三个部分。现在来分析并初步实现第一部分的功能。

1.1基于socket开发服务端流程

1.2需求分析/图解

工作:先打通自定义web服务器和浏览器之间的通道。

如浏览器请求http://localhost:8080/Xxx,服务器可以接收请求并返回简单数据。

注意:这里的交互是都建立在http协议之上的。服务器获取到的数据是http格式的,返回的数据也要封装成http格式,浏览器才能正常解析。

http格式详见javaweb-day14-HTTP协议

1.3代码实现

MyTomcatV1:

package com.li.MyTomcat;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @author 李
 * @version 1.0
 * 这是第一个版本的tomcat,可以完成接收浏览器请求,并返回信息功能
 */
public class MyTomcatV1 {
    public static void main(String[] args) throws IOException {

        //1.创建ServerSocket,在8080端口监听
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("==========mytomcat在8080端口监听=========");
        while (!serverSocket.isClosed()) {
            //等待浏览器或客户端的连接
            //如果有连接来,就创建socket
            //这个socket就是浏览器和服务器之间的连接(通道)
            Socket socket = serverSocket.accept();
            //先接收浏览器发送的数据
            InputStream inputStream = socket.getInputStream();//字节流
            //为了方便,将其转成字符流(InputStreamReader==>转换流,将一个字节流转换成字符流)
            //BufferedReader==>字符处理流
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "utf-8"));
            String mes = null;
            System.out.println("========接收到浏览器发送的数据=======");
            //循环地读取
            while ((mes = bufferedReader.readLine()) != null) {//按行读取数据,如果已到达流末尾,则返回 null
                //判断mes的长度是否为0
                if (mes.length() == 0) {
                    break;//退出while
                }
                System.out.println(mes);
            }

            //我们的tomcat回送数据-按照http格式

            //关闭资源
            inputStream.close();
            socket.close();

        }
    }
}

运行代码,在浏览器中发送请求http://localhost:8080/cal.html,后端输出如下:

可以看到,程序成功接收到了浏览器的请求(http格式)

下面以http格式响应浏览器请求:

package com.li.MyTomcat;

import java.io.*;
import java.net.ServerSocket;
import java.net.Socket;

/**
 * @author 李
 * @version 1.0
 * 这是第一个版本的tomcat,可以完成接收浏览器请求,并返回信息功能
 */
public class MyTomcatV1 {
    public static void main(String[] args) throws IOException {

            //1.创建ServerSocket,在8080端口监听
            /**
            * ..........
            */

            //mytomcat服务器回送数据-按照http格式
            OutputStream outputStream = socket.getOutputStream();
            //模仿响应头
            //\r\n表示回车换行:因为响应头最后一行和响应体中间要隔一个空行,
            // 所以最后一行要写两个\r\n(响应头换行是空行,空行再换行才是响应体)
            //即 http响应体前面需要有两个换行\r\n\r\n
            String respHeader = "HTTP/1.1 200 OK\r\n" +
                    "Content-Type: text/html;charset=utf-8\r\n\r\n";
            String resp = respHeader + "你好,我是服务器!";

            System.out.println("=====MyTomcat给浏览器回送的数据=====");
            System.out.println(resp);
            //注意:这里返回数据要以字节流方式返回
            outputStream.write(resp.getBytes());//将resp字符串以byte[]方式返回

            //关闭资源
            outputStream.flush();
            outputStream.close();
            inputStream.close();
            socket.close();
        }
    }
}

1.4测试

运行代码,在浏览器中发送请求http://localhost:8080/,后台输出如下:

浏览器输出如下:

2.实现任务阶段2-使用BIO线程模型,支持多线程

2.1BIO线程模型介绍

这里为了简单,使用方式一来完成操作,每次请求都会创建一个线程。

2.2需求分析/图解

  1. 需求分析如图所示,浏览器请求http://localhost:8080/,服务器返回“你好,我是服务器”。
  2. 后台mytomcat使用BIO线程模型,支持多线程,对数据的返回和处理移至线程里处理。

阶段1存在一个问题:当一个客户端被服务端在等待读取数据的时候,服务端会卡在那里,使得别的客户端无法与服务端连接以及收发数据。

解决方案是让等待接收数据的那块让子线程去做,主线程只需要一直监听是否有客户端连接即可,这样每个客户端就相互不影响了。

2.3代码实现

RequestHandler:

package com.li.MyTomcat.hander;

import java.io.*;
import java.net.Socket;

/**
 * @author 李
 * @version 1.0
 * RequestHandler是一个线程对象
 * 用来处理一个http请求
 */
public class RequestHandler implements Runnable {
    //定义一个Socket
    private Socket socket = null;

    //在创建RequestHandler对象的时候,将主线程的socket传给线程对象来使用
    public RequestHandler(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        //对客户端进行交互

        try {
            System.out.println("当前线程="+Thread.currentThread().getName());
            
            InputStream inputStream = socket.getInputStream();
            //将字节流转成字符流--方便按行读取浏览器请求
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "utf-8"));
            System.out.println("=====MyTomcat接收浏览器数据=====");
            String mes = null;
            while ((mes = bufferedReader.readLine()) != null) {
                //如果读取到的长度为零,即空串""
                if (mes.length() == 0) {
                    break;//退出循环
                }
                System.out.println(mes);
            }

            //将数据返回给浏览器/客户端(注意要将数据封装成http响应的格式,浏览器才能解析)
            //构建http响应头(注意换行-响应头和响应体之间有一个空行,需要两个换行)
            String respHeader = "HTTP/1.1 200 OK\r\n" +
                    "Content-Type: text/html;charset=utf-8\r\n\r\n";
            String resp = respHeader + "<h1>Hello,我是服务器!<h1>";
            System.out.println("=====MyTomcatV2给浏览器回送的数据=====");
            System.out.println(resp);
            //获取输出流
            OutputStream outputStream = socket.getOutputStream();
            outputStream.write(resp.getBytes());//将字符串转成字节数组

            //关闭流
            outputStream.flush();
            outputStream.close();
            inputStream.close();
            socket.close();

        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //一定要确保socket关闭
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

MyTomcatV2:

package com.li.MyTomcat;

import com.li.MyTomcat.hander.RequestHandler;

import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;

public class MyTomcatV2 {
    public static void main(String[] args) throws IOException {
        //监听
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("==========MyTomcatV2在8080端口监听=========");
        //只要serverSocket没有关闭,就会一直等待客户端来连接
        while (!serverSocket.isClosed()) {
            //每接收一个浏览器连接,就会得到一个socket对象(这个socket就是服务器和浏览器的数据通道)
            Socket socket = serverSocket.accept();
            //创建一个线程,将该socket传给该线程,启动线程
            new Thread(new RequestHandler(socket)).start();
        }
    }
}

2.4测试

运行代码,在浏览器输入http://localhost:8080/,输出如下:

后台输出:

3.实现任务阶段3-处理Servlet

  • Servlet声明周期

3.1需求分析/图解

问题分析:第二阶段只是简单地返回结果,没有和Servlet,web.xml关联起来

下面来完成第三阶段实现:

  • 自定义Servlet规范

要完成Servlet的调用就要先制定好Servlet规范,我们模仿Servlet的规范来制订一套自己的Servlet。

如下,模仿实际的Servlet来编写MyServlet及其抽象类MyHttpServlet,实现类MyCalServlet

  • 模拟实现request和response

另外,在真实的tomcat服务器中,接收http请求后会将其封装成HttpServletRequest对象,返回数据则是通过HttpServletResponse对象。

因此,除了自定义Servlet规范外,还要定义用于与http协议交互的两个对象

这里为了简化,就不使用实现接口的形式一层层封装了,直接用两个类来实现。

  • 在实现上面两个工作之前,先来看一个需求

我们现在做一个简单的需求:

浏览器请求http://localhost:8080/calServlet,提交数据,完成计算任务,如果该Servlet不存在,就返回404。大致的界面如下:

由于真正的HttpServletRequest和HttpServletResponse方法以及Servlet方法很多,这里为了简化,上述模拟的Servlet和自定义的request,response等只实现满足此需求的部分方法。

3.2模拟实现request和response

3.2.1MyRequest

package com.li.MyTomcat.http;

import java.io.BufferedReader;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.util.HashMap;

/**
 * @author 李
 * @version 1.0
 * 1.MyRequest的作用是封装http请求的数据
 * 2.比如 method,uri,还有参数列表等
 * 3.MyRequest 的作用相当于原生的Servlet中的HttpServletRequest
 * 4.这里先考虑get请求
 */
public class MyRequest {

    private String method;
    private String uri;
    //存放参数列表  参数名-参数值=>HashMap
    private HashMap<String, String> parametersMapping = new HashMap<>();
    private InputStream inputStream = null;

    //构造器==>对http请求进行封装
    //在构造 MyRequest对象的时候,将关联的socket的InputStream传进来
    // 这样就可以拿到该http请求的数据
    public MyRequest(InputStream inputStream) {
        this.inputStream = inputStream;
        packageHttp();
    }

    /**
     * 将http请求的相关参数进行封装,然后提供相关的方法,进行获取
     */
    public void packageHttp() {
        System.out.println("MyRequest packageHttp()被调用...");
        try {
            //将字节流转成字符流
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "utf-8"));
            /**
             * GET /myCalServlet?num1=11&num2=12 HTTP/1.1
             * Host: localhost:8080
             * User-Agent: Mozilla/5.0 (Windows NT 10.0; Win64; x64; rv:106.0) Gecko/20100101 Firefox/106.0
             * ......
             */
            //1.先读取第一行(请求行)
            String requestLine = bufferedReader.readLine();
            //第0组:GET  第1组:/myCalServlet?num1=11&num2=12 第2组:HTTP/1.1
            String[] requestLineArr = requestLine.split(" ");
            //2.得到method
            method = requestLineArr[0];
            //3.解析得到uri=/myCalServlet
            //先看看uri有没有参数列表
            int index = requestLineArr[1].indexOf("?");//找到?号的索引
            if (index == -1) {//没有问号,说明uri没有参数列表
                uri = requestLineArr[1];
            } else {
                //截取到?号的前一位,即为uri
                uri = requestLineArr[1].substring(0, index);
                //获取参数列表
                String parameters = requestLineArr[1].substring(index + 1);//parameters => num1=11&num2=12
                String[] parametersPair = parameters.split("&");//parametersPair=["num1=11","num2=12",...]
                //防止浏览器的提交的地址为 /myCalServlet?
                if (null != parametersPair && !"".equals(parametersPair)) {
                    //再分割
                    for (String parameterPair : parametersPair) {
                        //parameterVal=["num1","10"]
                        String[] parameterVal = parameterPair.split("=");
                        if (parameterVal.length == 2) {
                            //放入到parametersMapping中
                            parametersMapping.put(parameterVal[0], parameterVal[1]);
                        }
                    }
                }
            }
            //这里不能关闭流inputStream,因为inputStream和socket关联,
            // 如果关闭了inputStream则socket也会一起关闭
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    //request有一个特别重要的方法-getParameter()
    public String getParameter(String name) {
        if (parametersMapping.containsKey(name)) {
            return parametersMapping.get(name);
        } else {
            return "";
        }
    }

    public String getMethod() {
        return method;
    }

    public void setMethod(String method) {
        this.method = method;
    }

    public String getUri() {
        return uri;
    }

    public void setUri(String uri) {
        this.uri = uri;
    }

    @Override
    public String toString() {
        return "MyRequest{" +
                "method='" + method + '\'' +
                ", uri='" + uri + '\'' +
                ", parametersMapping=\n" + parametersMapping +
                '}';
    }
}

3.2.2MyResponse

package com.li.MyTomcat.http;

import java.io.OutputStream;

/**
 * @author 李
 * @version 1.0
 * 1.MyResponse可以封装OutputStream(和socket关联)
 * 2.即可以通过 MyResponse对象返回http响应给浏览器或客户端
 * 3.MyResponse的作用等价于原生的Servlet的HttpServletResponse
 */
public class MyResponse {
    private OutputStream outputStream = null;

    //写一个http的响应头
    public static final String respHeader = "HTTP/1.1 200 OK\r\n" +
            "Content-Type: text/html;charset=utf-8\r\n\r\n";

    //在构造 MyResponse 对象的时候,将关联的socket的OutputStream传进来
    public MyResponse(OutputStream outputStream) {
        this.outputStream = outputStream;
    }

    //当我们需要给浏览器返回数据时,可以通过MyResponse的属性获得输出流
    public OutputStream getOutputStream() {
        return outputStream;
    }
}

3.2.3修改RequestHandler

package com.li.MyTomcat.hander;

import com.li.MyTomcat.http.MyRequest;
import com.li.MyTomcat.http.MyResponse;

import java.io.*;
import java.net.Socket;

/**
 * @author 李
 * @version 1.0
 * RequestHandler是一个线程对象
 * 用来处理一个http请求
 */
public class RequestHandler implements Runnable {
    //定义一个Socket
    private Socket socket = null;

    //在创建RequestHandler对象的时候,将主线程的socket传给线程对象来使用
    public RequestHandler(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        //对客户端进行交互

        try {
            System.out.println("当前线程=" + Thread.currentThread().getName());

            InputStream inputStream = socket.getInputStream();

            MyRequest myRequest = new MyRequest(inputStream);
            String num1 = myRequest.getParameter("num1");
            String num2 = myRequest.getParameter("num2");
            String name = myRequest.getParameter("name");
            String email = myRequest.getParameter("email");
            System.out.println("请求的参数num1= " + num1);
            System.out.println("请求的参数num2= " + num2);
            System.out.println("请求的参数name= " + name);
            System.out.println("请求的参数email= " + email);
            System.out.println(myRequest);

            //这里我们可以通过myResponse对象返回数据给客户端
            MyResponse myResponse = new MyResponse(socket.getOutputStream());
            String resp = MyResponse.respHeader + "<h1>Hello,我是myResponse返回的信息</h1>";
            //这里的应用场景是:为了将来在Servlet中使用response对象,可以获取到输出流
            OutputStream outputStream = myResponse.getOutputStream();
            outputStream.write(resp.getBytes());
            outputStream.flush();
            outputStream.close();

            inputStream.close();
            socket.close();

        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            //一定要确保socket关闭
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }
}

MyTomcatV2类保持不变。

3.2.4测试

运行MyTomcatV2,在浏览器中输入如下请求:

http://localhost:8080/myCalServlet?num1=100&num2=200&name=jack&email=jack@qq.com

浏览器输出:

后台输出:

有关day02-实现01的更多相关文章

  1. ruby - 如何根据特征实现 FactoryGirl 的条件行为 - 2

    我有一个用户工厂。我希望默认情况下确认用户。但是鉴于unconfirmed特征,我不希望它们被确认。虽然我有一个基于实现细节而不是抽象的工作实现,但我想知道如何正确地做到这一点。factory:userdoafter(:create)do|user,evaluator|#unwantedimplementationdetailshereunlessFactoryGirl.factories[:user].defined_traits.map(&:name).include?(:unconfirmed)user.confirm!endendtrait:unconfirmeddoenden

  2. 华为OD机试用Python实现 -【明明的随机数】 2023Q1A - 2

    华为OD机试题本篇题目:明明的随机数题目输入描述输出描述:示例1输入输出说明代码编写思路最近更新的博客华为od2023|什么是华为od,od薪资待遇,od机试题清单华为OD机试真题大全,用Python解华为机试题|机试宝典【华为OD机试】全流程解析+经验分享,题型分享,防作弊指南华为o

  3. 基于C#实现简易绘图工具【100010177】 - 2

    C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.

  4. postman——集合——执行集合——测试脚本——pm对象简单示例02 - 2

    //1.验证返回状态码是否是200pm.test("Statuscodeis200",function(){pm.response.to.have.status(200);});//2.验证返回body内是否含有某个值pm.test("Bodymatchesstring",function(){pm.expect(pm.response.text()).to.include("string_you_want_to_search");});//3.验证某个返回值是否是100pm.test("Yourtestname",function(){varjsonData=pm.response.json

  5. MIMO-OFDM无线通信技术及MATLAB实现(1)无线信道:传播和衰落 - 2

     MIMO技术的优缺点优点通过下面三个增益来总体概括:阵列增益。阵列增益是指由于接收机通过对接收信号的相干合并而活得的平均SNR的提高。在发射机不知道信道信息的情况下,MIMO系统可以获得的阵列增益与接收天线数成正比复用增益。在采用空间复用方案的MIMO系统中,可以获得复用增益,即信道容量成倍增加。信道容量的增加与min(Nt,Nr)成正比分集增益。在采用空间分集方案的MIMO系统中,可以获得分集增益,即可靠性性能的改善。分集增益用独立衰落支路数来描述,即分集指数。在使用了空时编码的MIMO系统中,由于接收天线或发射天线之间的间距较远,可认为它们各自的大尺度衰落是相互独立的,因此分布式MIMO

  6. 【Java入门】使用Java实现文件夹的遍历 - 2

    遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg

  7. 牛客网专项练习30天Pytnon篇第02天 - 2

    1.在Python3中,下列关于数学运算结果正确的是:(B)a=10b=3print(a//b)print(a%b)print(a/b)A.3,3,3.3333...B.3,1,3.3333...C.3.3333...,3.3333...,3D.3.3333...,1,3.3333...解析:    在Python中,//表示地板除(向下取整),%表示取余,/表示除(Python2向下取整返回3)2.如下程序Python2会打印多少个数:(D)k=1000whilek>1:    print(k)k=k/2A.1000 B.10C.11D.9解析:    按照题意每次循环K/2,直到K值小于等

  8. ruby - Arrays Sets 和 SortedSets 在 Ruby 中是如何实现的 - 2

    通常,数组被实现为内存块,集合被实现为HashMap,有序集合被实现为跳跃列表。在Ruby中也是如此吗?我正在尝试从性能和内存占用方面评估Ruby中不同容器的使用情况 最佳答案 数组是Ruby核心库的一部分。每个Ruby实现都有自己的数组实现。Ruby语言规范只规定了Ruby数组的行为,并没有规定任何特定的实现策略。它甚至没有指定任何会强制或至少建议特定实现策略的性能约束。然而,大多数Rubyist对数组的性能特征有一些期望,这会迫使不符合它们的实现变得默默无闻,因为实际上没有人会使用它:插入、前置或追加以及删除元素的最坏情况步骤复

  9. ruby - "public/protected/private"方法是如何实现的,我该如何模拟它? - 2

    在ruby中,你可以这样做:classThingpublicdeff1puts"f1"endprivatedeff2puts"f2"endpublicdeff3puts"f3"endprivatedeff4puts"f4"endend现在f1和f3是公共(public)的,f2和f4是私有(private)的。内部发生了什么,允许您调用一个类方法,然后更改方法定义?我怎样才能实现相同的功能(表面上是创建我自己的java之类的注释)例如...classThingfundeff1puts"hey"endnotfundeff2puts"hey"endendfun和notfun将更改以下函数定

  10. ruby - 实现k最近邻需要哪些数据? - 2

    我目前有一个reddit克隆类型的网站。我正在尝试根据我的用户之前喜欢的帖子推荐帖子。看起来K最近邻或k均值是执行此操作的最佳方法。我似乎无法理解如何实际实现它。我看过一些数学公式(例如k表示维基百科页面),但它们对我来说并没有真正意义。有人可以推荐一些伪代码,或者可以查看的地方,以便我更好地了解如何执行此操作吗? 最佳答案 K最近邻(又名KNN)是一种分类算法。基本上,您采用包含N个项目的训练组并对它们进行分类。如何对它们进行分类完全取决于您的数据,以及您认为该数据的重要分类特征是什么。在您的示例中,这可能是帖子类别、谁发布了该项

随机推荐