JavaWeb 手写Tomcat底层机制

目录

一、Tomcat底层整体架构

        1.简介 : 

        2.分析图 : 

        3.基于Socket开发服务端的流程 : 

        4.打通服务器端和客户端的数据通道 : 

二、多线程模型的实现

        1.思路分析 : 

        2.处理HTTP请求 : 

        3.自定义Tomcat : 

三、自定义Servlet规范

        1. HTTP请求和响应 : 

            1° CyanServletRequest

            2° CyanServletResponse

        2.Servlet规范 : 

            1° CyanServlet

            2° CyanHttpServlet

            3° CyanLoginServlet

        3.容器实现 : 

            1° 思路分析

            2° web.xml配置文件

            3° 最终版自定义Tomcat

            4° 最终版自定义线程类

            5° 容器启动测试


一、Tomcat底层整体架构

        1.简介 : 

        Tomcat 有三种运行模式 (BIO[阻塞], NIO[非阻塞], APR),这里采用 BIO 线程模型来模拟实现。

        2.分析图 : 

                如下图所示 : 

                浏览器请求servlet资源后,Tomcat底层会通过Socket网络编程来接收请求,每次请求,都会创建一个新的线程去调用相应的Web资源,并返回。

        3.基于Socket开发服务端的流程 : 

                如下图所示 : 

                通过获取的Socket对象,来获取Socket对象对应的字节输入流和字节输出流。
                可以利用对象转换流将字节流转换为字符流对象(InputStreamReader实现了Reader抽象类),然后再通过BufferedReader的包装,将节点流转换成包装流(处理流)。

        4.打通服务器端和客户端的数据通道 : 

                PS : 
                Maven配置Web应用,运行出现jakarta.servlet.ServletException:
                因为tomcat10之后不是javax.servlet,而是jakarta.servlet,所以Web的依赖应该换成如下所示 : (pom.xml配置文件

<!--jar包的依赖-->
    <dependency>
      <groupId>jakarta.servlet</groupId>
      <artifactId>jakarta.servlet-api</artifactId>
      <version>5.0.0</version>
      <scope>provided</scope>
    </dependency>

<!--jsp的依赖-->
    <dependency>
      <groupId>jakarta.servlet.jsp</groupId>
      <artifactId>jakarta.servlet.jsp-api</artifactId>
      <version>3.0.0</version>
      <scope>provided</scope>
    </dependency>

                MyTomcat类代码如下 : (服务端;自定义的Tomcat)

package tomcat;

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

/**
    服务器端
 */
public class MyTomcat {
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("Tomcat在8080端口进行监听...");
        System.out.println("--------------------------------------");

        while (!serverSocket.isClosed()) {
            //获取Socket对象
            Socket socket = serverSocket.accept();

            //接收来自浏览器端的信息
            InputStream inputStream = socket.getInputStream();
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "UTF-8"));

            String content = null;  //局部变量在使用前必须赋初值。

            while ((content = bufferedReader.readLine()) != null) {
                //判断content是否是空串儿
                if (content.length() == 0) {
                    break;
                }
                System.out.println(content);
            }

            //服务器向浏览器回送消息
            OutputStream outputStream = socket.getOutputStream();
            //设置一个HTTP响应包的响应头
            // "/r/n"表示换行
            String respHeader =
                    "HTTP/1.1 200\r\n" +
                            "Content-Type: text/html;charset=utf-8\r\n\r\n";
            //设置HTTP响应的响应体
            String respBody = respHeader + "<h1>你好!</h1>";
            /*
                注意这里不能使用字符包装流来写数据!!!。
             */
            outputStream.write(respBody.getBytes());

            System.out.println("--------------------------------------");
            System.out.println(respBody);

            //释放资源
            outputStream.flush();
            outputStream.close();
            inputStream.close();
            socket.close();
//            serverSocket.close();
        }
    }
}

                在浏览器地址栏访问本机8080端口
                login.html代码如下 : 

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>login</title>
    <style>
        table, tr, td {
            border:2px cornflowerblue solid;
            border-collapse: collapse;
            padding: 10px;
            background-color: lightcyan
        }
        #tr01 {
            text-align: center
        }
    </style>
</head>
<body>
<form action="/Cyan_Tomcat/login" methods="get">
    <table>
        <tr>
            <th colspan="2">User Logging</th>
        </tr>
        <tr>
            <td>Username: </td>
            <td><input type="text" name="username"/></td>
        </tr>
        <tr>
            <td>Password: </td>
            <td><input type="password" name="password"/></td>
        </tr>
        <tr id="tr01">
            <td><input type="submit" value="Submit"/></td>
            <td><input type="reset" value="Reset"/></td>
        </tr>
    </table>
</form>
</body>
</html>

                LoginServlet类代码如下 : 

package servlet;

import jakarta.servlet.ServletException;
import jakarta.servlet.annotation.WebServlet;
import jakarta.servlet.http.HttpServlet;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.io.PrintWriter;

/**
 * @author : Cyan_RA9
 * @version : 21.0
 */
@WebServlet(urlPatterns = {"/login"})
public class LoginServlet extends HttpServlet {
    @Override
    protected void doPost(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        System.out.println("LoginServlet's doPost is invoked~");

        resp.setContentType("text/html; charset=utf-8");
        PrintWriter writer = resp.getWriter();

        req.setCharacterEncoding("utf-8");
        String username = req.getParameter("username");
        String password = req.getParameter("password");

        if ("Cyan".equals(username) && "123".equals(password)) {
            writer.print("<h1>登录成功!</h1>");
        } else {
            writer.print("<h1>登录失败!请重新尝试!</h1>");
        }
    }

    @Override
    protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
        doPost(req, resp);
    }
}

                运行效果 : 

                如果以Web工程配置好的Tomcat运行,就会按照LoginServlet类的代码逻辑来处理业务,如下图所示 : (GIF)

                如果以自定义的Tomcat运行,就会以MyTomcat类中的代码逻辑来处理业务。
                如下图所示 : (GIF)


二、多线程模型的实现

        1.思路分析 : 

                当服务器端接收到浏览器端的HTTP请求后,启动一个新的线程,令该线程持有浏览器对应的Socket对象,完成线程和浏览器的对接
                可通过实现Runnable接口的方式定义线程类HttpRequestHandler线程对象用于处理来自浏览器的HTTP请求

        2.处理HTTP请求 : 

                线程类HttpRequestHandler类代码如下 : 

package tomcat.handler;

import tomcat.http.CyanServletRequest;
import tomcat.http.CyanServletResponse;
import tomcat.servlet.CyanLoginServlet;

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

public class HttpRequestHandler implements Runnable {
    private Socket socket;

    public HttpRequestHandler(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        try {
            //接收客户端的信息
            InputStream inputStream = socket.getInputStream();

/*  以下代码已在CyanServletRequest类中实现
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "utf-8"));

            //BIO,每次请求都对应一个新的线程
            System.out.println("当前线程 = " + Thread.currentThread().getName());

            String content = null;  //局部变量在使用前必须赋初值。

            while ((content = bufferedReader.readLine()) != null) {
                //判断是否读到了空字符串
                if (content.length() == 0) {
                    break;
                }
                System.out.println(content);
            }
*/
            //Die first and live second
            //获取客户端的信息(利用了CyanServletRequest中封装好的方法)
            //以下代码已在CyanLoginServlet中实现
/*            CyanServletRequest cyanServletRequest = new CyanServletRequest(inputStream);
            String username = cyanServletRequest.getParameter("username");
            String password = cyanServletRequest.getParameter("password");
            System.out.println("username = " + username);
            System.out.println("password = " + password);
            System.out.println(cyanServletRequest);*/

            //给客户端回送信息
            //以下代码已在CyanLoginServlet类中实现。
/*            CyanServletResponse cyanServletResponse = new CyanServletResponse(socket.getOutputStream());
            String resp = CyanServletResponse.respHeader + "<h1>CyanServletResponse!</h1>";
            OutputStream outputStream = cyanServletResponse.getOutputStream();
            outputStream.write(resp.getBytes());
            outputStream.flush();
            outputStream.close();*/

            CyanServletRequest cyanServletRequest = new CyanServletRequest(inputStream);
            CyanServletResponse cyanServletResponse = new CyanServletResponse(socket.getOutputStream());

            CyanLoginServlet cyanLoginServlet = new CyanLoginServlet();
            cyanLoginServlet.doPost(cyanServletRequest, cyanServletResponse);

            //释放资源
            inputStream.close();
            socket.close();


/* 以下代码已在CyanServletResponse类中实现 :
            String respHeader = "HTTP/1.1 200\r\n" +
                    "Content-Type: text/html;charset=utf-8\r\n\r\n";
            String respHttp = respHeader + "<h1>Cyan_RA9</h1>";

            System.out.println("-----------------------------------------------");
            System.out.println("回送的信息如下:(回显)");
            System.out.println(respHttp);

            OutputStream outputStream = socket.getOutputStream();
            outputStream.write(respHttp.getBytes());

            //释放资源
            outputStream.flush();
            outputStream.close();
            inputStream.close();
            socket.close(); */
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            //确保Socket关闭
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }
}

        3.自定义Tomcat : 

                在MyTomcat_EX类中实现线程的分发
                MyTomcat_EX类代码如下 : 

package tomcat;

import tomcat.handler.HttpRequestHandler;

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

/**
 * @author : Cyan_RA9
 * @version : 21.0
 */
public class MyTomcat_EX {
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(8080);
        System.out.println("MyTomcat_EX在8080端口进行监听...");

        while (!serverSocket.isClosed()) {
            Socket socket = serverSocket.accept();
            HttpRequestHandler httpRequestHandler = new HttpRequestHandler(socket);
            Thread thread = new Thread(httpRequestHandler);

            thread.start();
        }
    }
}

                运行效果 : (GIF)


三、自定义Servlet规范

        1. HTTP请求和响应 : 

            1° CyanServletRequest

                CyanServletRequest类的作用等同于原始的HttpServletRequest,该类用于封装HTTP请求中的数据,eg : method, URI, 以及表单数据的参数列表等。
                CyanServletRequest类代码如下 : 

package tomcat.http;

import java.io.*;
import java.util.HashMap;

/**
 * @author : Cyan_RA9
 * @version : 21.0
 * @function : like the original HttpServletRequest.
 */
public class CyanServletRequest {
    private String method;
    private String URI;
    private HashMap<String, String> parametersMapping = new HashMap<>();
    private InputStream inputStream;

    /*
        此处传入的InputStream对象是和Socket关联的InputStream.
     */
    public CyanServletRequest(InputStream inputStream) {
        this.inputStream = inputStream;

        //完成对HTTP请求数据的封装
        this.init();
    }

    private void init() {
        System.out.println("\nCyanServletRequest's init is invoked~");
        try {
            //注意转换流的形参列表
            BufferedReader bufferedReader =
                    new BufferedReader(new InputStreamReader(inputStream, "utf-8"));

            //首先读取HTTP请求的请求行
            /* eg :  GET /Cyan/cyan.html HTTP/1.1 */
            String requestLine = bufferedReader.readLine();
            String[] requestLineArr = requestLine.split(" ");
            //获取method
            method = requestLineArr[0];
            //获取URI
            int index = requestLineArr[1].indexOf("?");
            if (index == -1) {      //if判断成立,说明请求行中没有参数列表
                URI = requestLineArr[1];
            } else {
                URI = requestLineArr[1].substring(0, index);
                //获取参数列表
                String parameters = requestLineArr[1].substring(index + 1);
                String[] parameterPairs = parameters.split("&");
                //兼容性处理,防止?后啥都没有。
                if (null != parameterPairs && !"".equals(parameterPairs)) {
                    for (String parameterPair : parameterPairs) {
                        String[] parameter = parameterPair.split("=");
                        if (parameter.length == 2) { //判断是否为一对完整的"name=value".
                            parametersMapping.put(parameter[0],parameter[1]);
                        }
                    }
                }
            }

            //!!! 直接关闭Socket关联的InputStream,会引起Socket的关闭。
            //inputStream.close();
        } catch (UnsupportedEncodingException e) {
            throw new RuntimeException(e);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    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;
    }

    //重要
    public String getParameter(String name) {
        if (parametersMapping.containsKey(name)) {      //注意此处API的使用!
            return parametersMapping.get(name);
        } else {
            return null;
        }
    }

    @Override
    public String toString() {
        return "CyanServletRequest{" +
                "method='" + method + '\'' +
                ", URI='" + URI + '\'' +
                ", parametersMapping=" + parametersMapping +
                '}';
    }
}

                CyanServletRequest类测试,运行效果如下GIF : 

            2° CyanServletResponse

                CyanServletResponse类的作用等同于原始的HttpServletResponse,用于封装HTTP响应的相关信息。
                CyanServletResponse类代码如下 : 

package tomcat.http;

import java.io.OutputStream;

/**
 * @author : Cyan_RA9
 * @version : 21.0
 * @function : like the original HttpServletResponse
 */
public class CyanServletResponse {
    private OutputStream outputStream;
    //设置一个HTTP响应头
    public static final String respHeader = "HTTP/1.1 200\r\n" +
            "Content-Type: text/html;charset=utf-8\r\n\r\n";

    //传入与Socket关联的OutputStream对象
    public CyanServletResponse(OutputStream outputStream) {
        this.outputStream = outputStream;
    }

    public OutputStream getOutputStream() {
        return outputStream;
    }
}

                运行测试(如下GIF图):

        2.Servlet规范 : 

            1° CyanServlet

                CyanServlet仅保留原生Servlet的init, destroy, service方法,其中,service方法供将来CyanServlet的抽象实现类CyanHttpServlet去重写。注意,service方法的形参列表,要使用自定义的CyanServletRequest 和 CyanServletResponse.
                CyanServlet接口,代码如下 : 

package tomcat.servlet;

import tomcat.http.CyanServletRequest;
import tomcat.http.CyanServletResponse;

import java.io.IOException;

public interface CyanServlet {
    void init() throws Exception;

    void service(CyanServletRequest req, CyanServletResponse resp) throws IOException;

    void destroy();
}

            2° CyanHttpServlet

                CyanHttpServlet的作用,类似于原生的HttpServlet;在CyanHttpServlet中实现CyanServlet接口中的service方法,在service方法中,要对HTTP请求的method类型进行判断。

                CyanHttpServlet抽象类代码如下 : 

package tomcat.servlet;

import tomcat.http.CyanServletRequest;
import tomcat.http.CyanServletResponse;

import java.io.IOException;

public abstract class CyanHttpServlet implements CyanServlet{
    @Override
    public void service(CyanServletRequest req, CyanServletResponse resp) throws IOException {
        //忽略大小写
        if ("GET".equalsIgnoreCase(req.getMethod())) {
            this.deGet(req, resp);
        } else if ("POST".equalsIgnoreCase(req.getMethod())) {
            this.doPost(req, resp);
        }
    }

    //模板设计模式
    public abstract void deGet(CyanServletRequest req, CyanServletResponse resp);
    public abstract void doPost(CyanServletRequest req, CyanServletResponse resp);
}

            3° CyanLoginServlet

                CyanLoginServlet是一个简单的servlet实例,用于继承CyanHttpServlet抽象类,并实现CyanHttpServlet类中的doGet和doPost抽象方法。之后,在HttpRequestHandler线程类中先尝试直接调用CyanLoginServlet实例。
                CyanLoginServlet代码如下 : 

package tomcat.servlet;

import tomcat.http.CyanServletRequest;
import tomcat.http.CyanServletResponse;

import java.io.IOException;
import java.io.OutputStream;

public class CyanLoginServlet extends CyanHttpServlet{
    @Override
    public void deGet(CyanServletRequest req, CyanServletResponse resp) {
        doPost(req, resp);
    }

    @Override
    public void doPost(CyanServletRequest req, CyanServletResponse resp) {
        String username = req.getParameter("username");
        String password = req.getParameter("password");

        //获取与当前Socket相关联的OutputStream对象
        OutputStream outputStream = resp.getOutputStream();
        String respInfo = CyanServletResponse.respHeader + "<h1>username = " +
                username + "</h1><br/> " + "<h1>password = " + password + "</h1>" + "<br/>" +
                "<h3>Cyan_RA9</h3>";
        try {
            outputStream.write(respInfo.getBytes());
            outputStream.flush();
            outputStream.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    @Override
    public void init() throws Exception {

    }

    @Override
    public void destroy() {

    }
}

                此外,还需要更新HttpServletHandler类中的类型,将已封装好的代码注释掉,HttpServletHandler类已更新,在上文“多线程模型实现”的HTTP请求处理模块。
                运行效果如下图所示 : 

        3.容器实现 : 

            1° 思路分析

                Tomcat中维护有至少两个大的HashMap容器。以web.xml配置文件的方式为例,其中一个HashMap容器,key存放<url-pattern>,value存放<servlet-name>;另一个HashMap容器,key存放<servlet-name>,value存放通过<servlet-class>来反射生成的servlet实例

            2° web.xml配置文件

                web.xml配置文件如下 : 

<!DOCTYPE web-app PUBLIC
 "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<!--xmlns="http://xmlns.jcp.org/xml/ns/javaee"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_4_0.xsd"
         version="4.0"-->
<!--IDEA报错——
  因为这是我们自定义的servlet,IDEA无法识别;无所谓!继续用!-->
<web-app>
  <display-name>Archetype Created Web Application</display-name>
  <servlet>
    <servlet-name>CyanLoginServlet</servlet-name>
    <servlet-class>tomcat.servlet.CyanLoginServlet</servlet-class>
  </servlet>
  <servlet-mapping>
    <servlet-name>CyanLoginServlet</servlet-name>
    <url-pattern>/cyanLogin</url-pattern>
  </servlet-mapping>
</web-app>

            3° 最终版自定义Tomcat

                在MyTomcat_Pro类中定义两个CurrentHashMap对象;定义init方法完成对两个CurrentHashMap对象的初始化(使用Dom4J读取web.xml配置文件)。
                首先,需要在Maven的pom.xml配置文件中,引入dom4j依赖,如下图所示 : 

                然后,将web.xml配置文件拷贝到/target/classes/目录下一份,如下图所示 : 

               MyTomcat_Pro代码如下 : 

package tomcat;

import org.dom4j.Document;
import org.dom4j.Element;
import org.dom4j.io.SAXReader;
import tomcat.handler.HttpRequestHandler;
import tomcat.servlet.CyanHttpServlet;

import java.io.File;
import java.io.IOException;
import java.lang.reflect.Constructor;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.List;
import java.util.concurrent.ConcurrentHashMap;

/**
 * The final own custom Tomcat.
 */
public class MyTomcat_Pro {
    //Tomcat维护的第一个容器
        //String --> <servlet-name>
        //CyanHttpServlet --> 可存放它的子类(即各种servlet实例)
    public static final ConcurrentHashMap<String, CyanHttpServlet> servletMapping =
            new ConcurrentHashMap<>();

    //Tomcat维护的第二个容器
        //String --> <url-patterns>
        //String --> <servlet-name>
    public static final ConcurrentHashMap<String, String> servletURLMapping =
            new ConcurrentHashMap<>();

    public static void main(String[] args) {
        MyTomcat_Pro myTomcat_pro = new MyTomcat_Pro();
        myTomcat_pro.init();
        myTomcat_pro.run();
    }

    public void run() {
        try {
            ServerSocket serverSocket = new ServerSocket(8080);
            System.out.println("MyTomcat_Pro在8080端口进行监听...");

            while (!serverSocket.isClosed()) {
                Socket socket = serverSocket.accept();
                Thread thread = new Thread(new HttpRequestHandler(socket));
                thread.start();
            }

        } catch (IOException e) {
            throw new RuntimeException(e);
        }
    }

    public void init() {
        String path = MyTomcat_Pro.class.getResource("/").getPath();
        //path = "/javaProject/Servlet/Cyan_Tomcat/target/classes/"

        //使用Dom4J技术解析web.xml文件
        SAXReader saxReader = new SAXReader();
        try {
            //注意文件名
            Document document = saxReader.read(new File(path + "web.xml"));
            System.out.println("document = " + document);
            //获取根元素<web-app>
            Element rootElement = document.getRootElement();
            //得到根元素下面的所有子元素
            List<Element> elements = rootElement.elements();
            //遍历并判断
            for (Element element : elements) {
                if ("servlet".equals(element.getName())) {
                    Element servlet_name = element.element("servlet-name");
                    Element servlet_class = element.element("servlet-class");

                    //反射机制创建servlet实例 (注意getText()方法的使用!)
                    Class<?> clazz = Class.forName(servlet_class.getText().trim());
                    Constructor<?> constructor = clazz.getConstructor();
                    CyanHttpServlet o = (CyanHttpServlet) constructor.newInstance();
                    servletMapping.put(servlet_name.getText(), o);
                } else if ("servlet-mapping".equals(element.getName())) {
                    Element url_pattern = element.element("url-pattern");
                    Element servlet_name = element.element("servlet-name");

                    servletURLMapping.put(url_pattern.getText(), servlet_name.getText());
                }
            }

        } catch (Exception e) {
            throw new RuntimeException(e);
        }
    }
}

            4° 最终版自定义线程类

package tomcat.handler;

import tomcat.MyTomcat_Pro;
import tomcat.http.CyanServletRequest;
import tomcat.http.CyanServletResponse;
import tomcat.servlet.CyanHttpServlet;
import tomcat.servlet.CyanLoginServlet;

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

public class HttpRequestHandler implements Runnable {
    private Socket socket;

    public HttpRequestHandler(Socket socket) {
        this.socket = socket;
    }

    @Override
    public void run() {
        try {
            //接收客户端的信息
            CyanServletRequest cyanServletRequest = new CyanServletRequest(socket.getInputStream());
            CyanServletResponse cyanServletResponse = new CyanServletResponse(socket.getOutputStream());

            String uri = cyanServletRequest.getURI();
            System.out.println("uri = " + uri);
            String servlet_name = MyTomcat_Pro.servletURLMapping.get(uri);
            /*
                这里的servlet_name可能为空。
                解决方式一 : 将CurrentHashMap替换为HashMap
                解决方式二 : 增加一个是否为null的判断。
             */
            if (servlet_name == null) {
                servlet_name = "";
            }

            //多态 --> 动态绑定
            CyanHttpServlet cyanHttpServlet = MyTomcat_Pro.servletMapping.get(servlet_name);
            if (cyanHttpServlet != null) {  //判断是否正常得到servlet实例
                cyanHttpServlet.service(cyanServletRequest, cyanServletResponse);
            } else {
                //如果没有找到servlet,返回404
                String resp = CyanServletResponse.respHeader + "<h1>404 Not Found!</h1>";
                OutputStream outputStream = cyanServletResponse.getOutputStream();
                outputStream.write(resp.getBytes());
                outputStream.flush();
                outputStream.close();
            }

            //释放资源
            socket.close();
        } catch (IOException e) {
            throw new RuntimeException(e);
        } finally {
            //确保Socket关闭
            if (socket != null) {
                try {
                    socket.close();
                } catch (IOException e) {
                    throw new RuntimeException(e);
                }
            }
        }
    }
}

            5° 容器启动测试

                如下图所示(GIF):

 

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

成集云 | 畅捷通采购单同步至钉钉 | 解决方案

源系统成集云目标系统 介绍 畅捷通是一家专业的金融科技公司&#xff0c;致力于为投资者提供便捷、高效的金融服务。通过畅捷通T的交易方式&#xff0c;投资者可以更加灵活地进行买卖交易&#xff0c;并且在交易完成后即可获得结算款项&#xff0c;无需等待T1的结算周期。 钉…

Styletron: 面向组件的样式设计工具包

styletron官网&#xff1a; styletron的GitHub链接&#xff1a; styletron-react 一. 介绍 Styletron是一个通用的component-oriented&#xff08;面向组件的&#xff09;样式工具。它属于css-in-js类别。Styletron可以很好地与React配合使用&#xff0c;但也可以与其他框架或…

Python弹球小游戏

给在校的小妹妹做个游戏玩&#xff1a;. 弹珠游戏主要是靠坐标xy&#xff0c;接板长度&#xff0c;球的半径等决定&#xff1a; # -*- coding: utf-8 -*- # Author : Codeooo # Time : 2022/04/29import sys import time import random import pygame as pgprint("&q…

ML类CFAR检测器在不同环境中检测性能的分析

摘要&#xff1a;该文是楼主翻阅书籍以及一些论文总结出来的关于ML(均值)类CFAR检测器在不同环境中的性能对比&#xff0c;以及优缺点的总结&#xff0c;可以帮助大家面对不同情形如何选择CFAR问题。由于楼主见识短浅&#xff0c;文中难免出现不足之处&#xff0c;望各位指出。…

校对软件助力公安公检:提高调查报告质量

校对软件可以为公安公检机关提供有力支持&#xff0c;帮助提高调查报告的质量。以下是校对软件在这方面的助力&#xff1a; 1.拼写和语法检查&#xff1a;校对软件可以自动检查调查报告中的拼写错误和语法问题。这可以避免由于疏忽或拼写错误而导致的报告不准确或难以理解的情况…

uniapp scroll-view 隐藏滚动条

/*清除滚动条 - 适配安卓*/::-webkit-scrollbar {width: 0;height: 0;color: transparent;}/*清除滚动条 - 适配IOS*/::-webkit-scrollbar {display: none;}

对强缓存和协商缓存的理解

浏览器缓存的定义&#xff1a; 浏览器缓存是浏览器在本地磁盘对用户最近请求过的文档进行存储&#xff0c;当访问者再次访问同一页面时&#xff0c;浏览器就可以直接从本地磁盘加载文档。 浏览器缓存分为强缓存和协商缓存。 浏览器是如何使用缓存的&#xff1a; 浏览器缓存…

oracle的管道函数

Oracle管道函数(Pipelined Table Function)oracle管道函数 1、管道函数即是可以返回行集合&#xff08;可以使嵌套表nested table 或数组 varray&#xff09;的函数&#xff0c;我们可以像查询物理表一样查询它或者将其赋值给集合变量。 2、管道函数为并行执行&#xff0c;在…

传统图像算法 - 运动目标检测之KNN运动背景分割算法

以下代码用OpenCV实现了视频中背景消除和提取的建模&#xff0c;涉及到KNN&#xff08;K近邻算法&#xff09;&#xff0c;整体效果比较好&#xff0c;可以用来进行运动状态分析。 原理如下&#xff1a; 背景建模&#xff1a;在背景分割的开始阶段&#xff0c;建立背景模型。 …

【ChatGPT 指令大全】怎么使用ChatGPT来辅助学习英语

在当今全球化的社会中&#xff0c;英语已成为一门世界性的语言&#xff0c;掌握良好的英语技能对个人和职业发展至关重要。而借助人工智能的力量&#xff0c;ChatGPT为学习者提供了一个有价值的工具&#xff0c;可以在学习过程中提供即时的帮助和反馈。在本文中&#xff0c;我们…

力扣63.不同路径II(动态规划)

/*** author Limg* date 2022/08/09* 一个机器人位于一个 m x n 网格的左上角 &#xff08;起始点在下图中标记为 “Start” &#xff09;。* 机器人每次只能向下或者向右移动一步。机器人试图达到网格的右下角&#xff08;在下图中标记为 “Finish”&#xff09;。* 现在考虑网…

国产低功耗蓝牙HS6621CxC/6621Px系列支持Find My网络功能方案芯片

目录 什么是“Find My“&#xff1f;HS6621系列简介 什么是“Find My“&#xff1f; “Find My”是苹果公司于19年前推出的针对失物追踪&#xff0c;Find My iPhone&#xff08;查找我的iPhone&#xff09;和Find My Friends&#xff08;查找朋友&#xff09;的结合体应用。为…

oracle容灾备份怎么样Oracle容灾备份

随着科学技术的发展和业务的增长&#xff0c;数据安全问题越来越突出。为了保证数据的完整性、易用性和保密性&#xff0c;公司需要采取一系列措施来防止内容丢失的风险。  Oracle是一个关系数据库管理系统(RDBMS),OracleCorporation是由美国软件公司开发和维护的。该系统功能…

网络编程——数据包的组装和拆解

数据包的组装和拆解 一、数据包在各个层之间的传输 二、各个层的封包格式 1、链路层封包格式 -------------------------------------------------------------------------------------------------------------------------------------- | 目标MAC地址&#xff08;6字节&a…

干货分享|Elsevier投稿进度查询功能正式上线,随时获取投稿状态!

想必广大科研学者们都经历过每天登录系统查看投稿进度的煎熬过程&#xff0c;为了方便广大科研人随时获取投稿状态&#xff0c;2023年8月&#xff0c;Elsevier【微信端投稿进度查询功能】正式上线&#xff01; 无论你是通讯作者还是共同作者&#xff0c;只需一次查询&#xff…

前端先行模拟接口(mock+expres+json)

目录 mock模拟数据&#xff1a;data/static.js 路由&#xff1a;index.js 服务器&#xff1a;server.js yarn /node 启动服务器&#xff1a;yarn start 客户端&#xff1a;修改代理路径(修改设置后都要重启才生效) 示例 后端框架express构建服务器 前端发起请求 静态数…

ElastAlert通过飞书机器人发送报警通知

前言 公司采用ELK架构搜集业务系统的运行日志&#xff0c;以前开发人员只有在业务出现问题的时候&#xff0c;才会去kibana上进行日志搜索操作&#xff0c;每次都是被用户告知系统出问题了&#xff0c;这简直是被啪啪打脸~ 于是痛定思痛&#xff0c;决定主动出击&#xff0c;…

探索泛型与数据结构:解锁高效编程之道

文章目录 引言第一部分&#xff1a;了解泛型1.1 为什么使用泛型1.2 使用泛型的好处 第二部分&#xff1a;泛型的使用场景2.1 类的泛型2.2 方法的泛型2.3 接口的泛型 第三部分&#xff1a;泛型通配符3.1 通配符3.2 通配符的受限泛型 第四部分&#xff1a;数据结构和泛型的应用4.…

Jenkins 中 shell 脚本执行失败却不自行退出

Jenkins 中 执行 shell 脚本时&#xff0c;有时候 shell 执行失败了&#xff0c;或者判断结果是错误的&#xff0c;但是 Jenkins 执行完成后确提示成功 success 。 此时&#xff0c;可以通过条件判断来解决这个问题&#xff0c;让 Jenkins 强制退出并提示执行失败 failed 。 …

40% Ubuntu 用户面临着新特权提升漏洞风险

导读Wiz 的研究人员发现&#xff0c;最近被引入 Ubuntu 内核的两个 Linux 漏洞&#xff0c;可能会在大量设备上为非特权本地用户提升权限。这两个漏洞被追踪为 CVE-2023-32629 和 CVE-2023-2640&#xff0c;预计影响了大约 40% 的 Ubuntu 用户。 根据介绍&#xff0c;其中 CVE…