how tomcat work 搬运工(一)

charter 1

基本的服务端接收请求并处理

包和类

HttpServer  是主函数入口,通过await()方法新建ServerSock 对象实例来绑定端口,接收请求,通过请求来新建 Request 和 Response 对象实例,

package com.howtomcatwork.xy;

import java.net.Socket;
import java.net.ServerSocket;
import java.net.InetAddress;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.IOException;
import java.io.File;

public class HttpServer {

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

    // shutdown command
    private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";

    // the shutdown command received
    private boolean shutdown = false;

    public static void main(String[] args) {
        HttpServer server = new HttpServer();
        server.await();
    }

    public void await() {
        ServerSocket serverSocket = null;
        int port = 8080;
        try {
            serverSocket = new ServerSocket(port, 1,
                    InetAddress.getByName("127.0.0.1"));
        } catch (IOException e) {
            e.printStackTrace();
            System.exit(1);
        }

        // Loop waiting for a request
        while (!shutdown) {
            Socket socket = null;
            InputStream input = null;
            OutputStream output = null;
            try {
                socket = serverSocket.accept();
                input = socket.getInputStream();
                output = socket.getOutputStream();

                // create Request object and parse
                Request request = new Request(input);
                request.parse();

                // create Response object
                Response response = new Response(output);
                response.setRequest(request);
                response.sendStaticResource();

                // Close the socket and send outputstream to client
                socket.close();

                // check if the previous URI is a shutdown command
                shutdown = request.getUri().equals(SHUTDOWN_COMMAND);
            } catch (Exception e) {
                e.printStackTrace();
                continue;
            }
        }
    }
}

自定义的Request 类

自定义的Request 类主要接收 inputStream(接收的请求数据)的2进制数据如何进行处理,

parse()函数将inputstream转换为string,并通过parseUri函数将uri赋值,

parseUri(String )函数将String 类型的请求中获取请求的url,

package com.howtomcatwork.xy;

import java.io.InputStream;
import java.io.IOException;

public class Request {
    private InputStream input;
    private String uri;

    public Request(InputStream input) {
        this.input = input;
    }

    public void parse() {
        // Read a set of characters from the socket
        StringBuffer request = new StringBuffer(2048);
        int i;
        byte[] buffer = new byte[2048];
        try {
            i = input.read(buffer);
        } catch (IOException e) {
            e.printStackTrace();
            i = -1;
        }
        for (int j = 0; j < i; j++) {
            request.append((char) buffer[j]);
        }
        System.out.print(request.toString());
        uri = parseUri(request.toString());
    }

    private String parseUri(String requestString) {
        int index1, index2;
        index1 = requestString.indexOf(‘ ‘);
        if (index1 != -1) {
            index2 = requestString.indexOf(‘ ‘, index1 + 1);
            if (index2 > index1)
                return requestString.substring(index1 + 1, index2);
        }
        return null;
    }

    public String getUri() {
        return uri;
    }

}

自定义的Response 类

sendStaticResource() 通过请求的uri  读取文件或返回404提示,将读取的内容或404提示的bit形式输出到响应的输出流

package com.howtomcatwork.xy;

import java.io.OutputStream;
import java.io.IOException;
import java.io.FileInputStream;
import java.io.File;

public class Response {

    private static final int BUFFER_SIZE = 1024;
    Request request;
    OutputStream output;

    public Response(OutputStream output) {
        this.output = output;
    }

    public void setRequest(Request request) {
        this.request = request;
    }

    public void sendStaticResource() throws IOException {
        byte[] bytes = new byte[BUFFER_SIZE];
        FileInputStream fis = null;
        try {
            File file = new File(HttpServer.WEB_ROOT, request.getUri());
            if (file.exists()) {
                fis = new FileInputStream(file);
                int ch = fis.read(bytes, 0, BUFFER_SIZE);
                while (ch != -1) {
                    output.write(bytes, 0, ch);
                    ch = fis.read(bytes, 0, BUFFER_SIZE);
                }
            } else {
                // file not found
                String errorMessage = "HTTP/1.1 404 File Not Found\r\n"
                        + "Content-Type: text/html\r\n"
                        + "Content-Length: 23\r\n" + "\r\n"
                        + "<h1>File Not Found</h1>";
                output.write(errorMessage.getBytes());
            }
        } catch (Exception e) {
            // thrown if cannot instantiate a File object
            System.out.println(e.toString());
        } finally {
            if (fis != null)
                fis.close();
        }
    }
}

完整的目录结构

主要参考  https://www.gitbook.com/book/l-webx/how_tomcat_works/details

时间: 2024-11-08 08:54:54

how tomcat work 搬运工(一)的相关文章

《how tomcat work》 搬运工 Chapter 15: Digester

在之前的章节都是通过Bootstrap类来初始化connector,context,wrapper. 而且手动来绑定它们的关系. connector.setContainer(context); 或者手动设置实例的属性 context.setPath("/myApp") ; context.setDocBase("myApp"); 在tomcat中,是用web.xml来设置的.而digester是一个开源的库,可以读取xml,通过xml来加载类,实例化. Diges

《how tomcat work》 搬运工 charpter 3:Connector

这一章节主要是介绍了tomcat 的一个重要模块 connector,但只是一个缩减版 connector的作用是创建ServerSocket的实例serverSocket,绑定端口,监听端口,当接受到客户端请求时,就创建一个HttpProcessor实例processor,将serverSocket作为参数给HttpProcessor的实例processor.process(serverSocket)来处理请求 import java.io.IOException; import java.n

《how tomcat work》 搬运工 Chapter 14 Server and Service

之前的章节是将container和connector联系了在一起,但是connector只能和一个端口相连接.而且,之前的application都缺少了一个开启服务,关闭服务的接口.这章节就是介绍tomcat的server和service.server和service就是方便了启动. Server  org.apache.catalina.Server接口代表了catalinaservlet的server.server整合了将connector和container的开启和关闭. public i

《how tomcat work》 搬运工 Chapter 11:StandardWrapper

tomcat 4和5中有四种container,engine,host,context,wrapper,之前已经简单介绍过自己编写的simpleContext,simpleWrapper.这章节就是介绍catalina的StandardWrapper. 首先wrapper是代表一个servlet,所以,wrapper的作用是,servlet类的加载,servlet的信息,还有进入servlet的fiter的信息,还有fiter的加载,fiter的chain. 1.servlet可以继承javax

《how tomcat work》 搬运工 Chapter 9: Session Management

Catalina版本的Tomcat通过manager板块来提供session管理,manager通常是和context联系在一起,就是context类里会有一个manager实例. Session interface public interface Session { public static final String SESSION_CREATED_EVENT = "createSession"; public static final String SESSION_DESTRO

《how tomcat work》 搬运工 charpter2: A Simple Servlet Container

这一章介绍了servlet container,application中对应的类是主要是httpServer1,这个类是创建socket绑定端口,接受请求,然后创建request实例和response实例,根据request的uri来处理请求,如果是静态资源就创建StaticResourceProcessor实例来处理,如果是servlet请求则创建对应servlet来处理请求 这章节的request和response类继承了servletRequest接口,所以对应的pom要添加依赖项 <de

《how tomcat work》 搬运工 Chapter 5: Container

Containner的作用是将请求送到servlet中,然后产生response给客户端.在tomcat 4中,container有四种:Enginer,Host,Context,和 Wrapper.这个章节主要是简单介绍下Context和Wrapper,其中的Context和Wrapper都是自定义比较简单的. Engine:代表整个Catalina servlet的引擎. Host:代表一个拥有多个context的虚拟主机. Context:代表一个网站,一个Context拥有一个或多个Wr

《how tomcat work》 搬运工 charpter 4:Default Connector

Default Connector是Tomcat 4用的connector Default Connector的工作是等待HTTP请求,创建request和response对象,然后传给container,然后container加载servlet来进行处理 Default Connector和charpter 3里的Connector区别是: 1.connector的HttpProcessor是同步的,要等到一个请求结束后才可以处理下一个请求.而Default Connector的HttpPro

解决myeclipse中tomcat启动报错A configuration error occurred during startup. Please verify the preference field whit the prompt:Cannot connect to VM

原因一:可能是tomcat里面文件不小心被删除了.尝试更换tomcat, 解决步骤:重新下载tomcat并解压.点击windows——> preferences——>tomcat重新配置. 原因二:myeclipse系统自带jdk出问题 解决步骤:重新下载tomcat并解压.点击windows——> preferences——>tomcat——>jdk,配置成你自己的jdk 原因三:由于软件冲突导致的错误 解决步骤: 找到myeclipse.ini文件 加上-Djava.ne