采坑-深入剖析Tomcat一书 第二章

项目结构:

贴代码:

Constants:
package com.diy.tomcat.chapter2.servlet;

import java.io.File;

public class Constants {
    /**
     *  WEB_ROOT是HTML和其他文件所在的目录。对于这个包,WEB_ROOT是工作目录下的“webroot”目录。工作目录是文件系统中调用java命令的位置。
     */
    public static final String WEB_ROOT = System.getProperty("user.dir") + File.separator + "webroot/production/";
}
HttpServer1:
package com.diy.tomcat.chapter2.servlet;

import com.diy.tomcat.server.Request;

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

public class HttpServer1 {


    private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";

    private boolean shutdown = false;

    public static void main(String[] args) {
        HttpServer1 httpServer1 = new HttpServer1();
        httpServer1.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);
        }

        while (!shutdown) {
            Socket socket = null;
            InputStream inputStream = null;
            OutputStream outputStream = null;
            try {
                socket = serverSocket.accept();
                inputStream = socket.getInputStream();
                outputStream = socket.getOutputStream();

                Request1 request1 = new Request1(inputStream);
                request1.parse();

                Response1 response1 = new Response1(outputStream);
                response1.setRequest(request1);

                if (request1.getUri().startsWith("/servlet")) {
                    ServletProcessor1 processor1 = new ServletProcessor1();
                    processor1.process(request1, response1);
                } else {
                    StaticResourceProcessor processor = new StaticResourceProcessor();
                    processor.process(request1, response1);
                }

                socket.close();

                shutdown = request1.getUri().equals(SHUTDOWN_COMMAND);

            } catch (Exception e) {
                e.printStackTrace();
                System.exit(1);
            }

        }



    }

}
PrimitiveServlet:
package com.diy.tomcat.chapter2.servlet;

import javax.servlet.*;
import java.io.IOException;
import java.io.PrintWriter;

public class PrimitiveServlet implements Servlet {
    @Override
    public void init(ServletConfig servletConfig) throws ServletException {
        System.out.println("这是自己写的PrimitiveServlet的初始化方法");
    }

    @Override
    public ServletConfig getServletConfig() {
        return null;
    }

    @Override
    public void service(ServletRequest servletRequest, ServletResponse servletResponse) throws ServletException, IOException {
        System.out.println("这是PrimitiveServlet中可能会被调用多次的service方法。");

        PrintWriter out = servletResponse.getWriter();
        StringBuffer sb = new StringBuffer();
        //响应头信息
        sb.append("HTTP/1.1 200 OK\n");
        sb.append("Content-Type: text/html;charset=UTF-8\n");
        sb.append("\r\n");
        sb.append("<h1>Hello. Rose are red </h1>");

        out.println(sb.toString());
        out.println("Hello. Rose are red.!!!!!!!!!!!");
        out.print("Violets are blue");
    }

    @Override
    public String getServletInfo() {
        return null;
    }

    @Override
    public void destroy() {
        System.out.println("这是PrimitiveServlet的destroy方法。用以关闭servlet容器和释放内存。");
    }
}
Request1
package com.diy.tomcat.chapter2.servlet;

import javax.servlet.*;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.util.Enumeration;
import java.util.Locale;
import java.util.Map;

public class Request1 implements ServletRequest {

    private InputStream inputStream;
    private String uri;

    public Request1(InputStream inputStream) {
        this.inputStream = inputStream;
    }

    public String getUri() {
        return uri;
    }

    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 void parse() {
        StringBuffer stringBuffer = new StringBuffer(2048);
        int i;
        byte[] buffer = new byte[2048];
        try {
            i = inputStream.read(buffer);
        } catch (IOException e) {
            e.printStackTrace();
            i = -1;
        }
        for (int j = 0; j < i; j++) {
            stringBuffer.append((char) buffer[j]);
        }
        System.out.println(stringBuffer.toString());
        uri = parseUri(stringBuffer.toString());
    }


    @Override
    public Object getAttribute(String s) {
        return null;
    }

    @Override
    public Enumeration<String> getAttributeNames() {
        return null;
    }

    @Override
    public String getCharacterEncoding() {
        return null;
    }

    @Override
    public void setCharacterEncoding(String s) throws UnsupportedEncodingException {

    }

    @Override
    public int getContentLength() {
        return 0;
    }

    @Override
    public String getContentType() {
        return null;
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {
        return null;
    }

    @Override
    public String getParameter(String s) {
        return null;
    }

    @Override
    public Enumeration<String> getParameterNames() {
        return null;
    }

    @Override
    public String[] getParameterValues(String s) {
        return new String[0];
    }

    @Override
    public Map<String, String[]> getParameterMap() {
        return null;
    }

    @Override
    public String getProtocol() {
        return null;
    }

    @Override
    public String getScheme() {
        return null;
    }

    @Override
    public String getServerName() {
        return null;
    }

    @Override
    public int getServerPort() {
        return 0;
    }

    @Override
    public BufferedReader getReader() throws IOException {
        return null;
    }

    @Override
    public String getRemoteAddr() {
        return null;
    }

    @Override
    public String getRemoteHost() {
        return null;
    }

    @Override
    public void setAttribute(String s, Object o) {

    }

    @Override
    public void removeAttribute(String s) {

    }

    @Override
    public Locale getLocale() {
        return null;
    }

    @Override
    public Enumeration<Locale> getLocales() {
        return null;
    }

    @Override
    public boolean isSecure() {
        return false;
    }

    @Override
    public RequestDispatcher getRequestDispatcher(String s) {
        return null;
    }

    /**
     * @param s
     * @deprecated
     */
    @Override
    public String getRealPath(String s) {
        return null;
    }

    @Override
    public int getRemotePort() {
        return 0;
    }

    @Override
    public String getLocalName() {
        return null;
    }

    @Override
    public String getLocalAddr() {
        return null;
    }

    @Override
    public int getLocalPort() {
        return 0;
    }

    @Override
    public ServletContext getServletContext() {
        return null;
    }

    @Override
    public AsyncContext startAsync() {
        return null;
    }

    @Override
    public AsyncContext startAsync(ServletRequest servletRequest, ServletResponse servletResponse) {
        return null;
    }

    @Override
    public boolean isAsyncStarted() {
        return false;
    }

    @Override
    public boolean isAsyncSupported() {
        return false;
    }

    @Override
    public AsyncContext getAsyncContext() {
        return null;
    }

    @Override
    public DispatcherType getDispatcherType() {
        return null;
    }
}
Response1
package com.diy.tomcat.chapter2.servlet;

import com.diy.tomcat.server.HttpServer;

import javax.servlet.ServletOutputStream;
import javax.servlet.ServletResponse;
import java.io.*;
import java.util.Locale;

public class Response1 implements ServletResponse {

    private static final int BUFFER_SIZE = 1024;
    Request1 request1;
    OutputStream outputStream;
    PrintWriter writer;

    public Response1(OutputStream outputStream) {
        this.outputStream = outputStream;
    }

    public void setRequest(Request1 request1) {
        this.request1 = request1;
    }

    public void sendStaticResource() throws IOException {
        byte[] bytes = new byte[BUFFER_SIZE];
        FileInputStream fileInputStream = null;
        try {
            File file = new File(HttpServer.WEB_ROOT, request1.getUri());

            if (file.exists()) {
                fileInputStream = new FileInputStream(file);
                int ch = fileInputStream.read(bytes, 0, BUFFER_SIZE);

                StringBuffer sb = new StringBuffer();
                //响应头信息
                sb.append("HTTP/1.1 200 OK\n");
                sb.append("Content-Type: text/html;charset=UTF-8\n");
                sb.append("\r\n");
                while (ch != -1) {
                    sb.append(new String(bytes));
                    outputStream.write(sb.toString().getBytes());
//                    outputStream.write(bytes, 0, ch);
                    ch = fileInputStream.read(bytes, 0, BUFFER_SIZE);
                }
            } else {
                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>";
                outputStream.write(errorMessage.getBytes());
            }
        } catch (Exception e) {
            e.printStackTrace();
            // 如果不能实例化文件对象,则抛出
            System.out.println(e.toString());
        } finally {
            if (fileInputStream != null) {
                fileInputStream.close();
                outputStream.flush();
                outputStream.close();
            }
        }
    }


    @Override
    public String getCharacterEncoding() {
        return null;
    }

    @Override
    public String getContentType() {
        return null;
    }

    @Override
    public ServletOutputStream getOutputStream() throws IOException {
        return null;
    }

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

    @Override
    public void setCharacterEncoding(String s) {

    }

    @Override
    public void setContentLength(int i) {

    }

    @Override
    public void setContentType(String s) {

    }

    @Override
    public void setBufferSize(int i) {

    }

    @Override
    public int getBufferSize() {
        return 0;
    }

    @Override
    public void flushBuffer() throws IOException {

    }

    @Override
    public void resetBuffer() {

    }

    @Override
    public boolean isCommitted() {
        return false;
    }

    @Override
    public void reset() {

    }

    @Override
    public void setLocale(Locale locale) {

    }

    @Override
    public Locale getLocale() {
        return null;
    }
}
ServletProcessor1
package com.diy.tomcat.chapter2.servlet;


import javax.servlet.Servlet;
import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import java.io.File;
import java.io.IOException;
import java.net.URL;
import java.net.URLClassLoader;

public class ServletProcessor1 {

    public void process (Request1 request1, Response1 response1) {
        String uri = request1.getUri();
        String servletName = uri.substring(uri.lastIndexOf("/") + 1);
        System.out.println("=="+ servletName);
        URLClassLoader urlClassLoader = null;
        try {
            // 创建一个URLClassLoader
//            URL[] urls = new URL[1];
//            URLStreamHandler urlStreamHandler = null;
//            File classPath = new File(Constants.WEB_ROOT);

//            // 存储库的形成是从createClassLoader方法中获得的组织 阿帕奇.卡特琳娜.startup.ClassLoader工厂
//            String repository = (new URL("file", null, classPath.getCanonicalPath() + File.separator)).toString();
//
//            System.out.println("===" + repository);
//
//            // 形成URL的代码取自addRepository方法阿帕奇.卡特琳娜.装载机。默认类加载器。
//            urls[0] = new URL(null, repository, urlStreamHandler);
//
//            urlClassLoader = new URLClassLoader(urls);
            File f = new File(this.getClass().getResource("/").getPath());
            String path = "file:" + f.toString();
            URL url = new URL(path);
            urlClassLoader = new URLClassLoader(new URL[]{url});

        }catch (IOException e) {
            e.printStackTrace();
            System.out.println(e.toString());
        }

        String packagePath = "com.diy.tomcat.chapter2.servlet.";

        servletName = packagePath + servletName;

        Class myClass = null;
        try {
            myClass = urlClassLoader.loadClass(servletName);
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
            System.out.println(e.toString());
        }

        Servlet servlet = null;

        try {
            servlet = (Servlet) myClass.newInstance();
//            servlet.init(servlet.getServletConfig());
            servlet.service((ServletRequest) request1, (ServletResponse) response1) ;
//            servlet.destroy();
        } catch (Exception e) {
            e.printStackTrace();
            System.out.println(e.toString());
        }catch (Throwable e) {
            e.printStackTrace();
        }

    }
}
StaticResourceProcessor
package com.diy.tomcat.chapter2.servlet;

import java.io.IOException;

public class StaticResourceProcessor {

    public void process(Request1 request1, Response1 response1) {
        try {
            response1.sendStaticResource();
        } catch (IOException e) {
            e.printStackTrace();
        }

    }
}

其中碰到的坑有两个,一个就是返回信息,response.getWriter(),要跟单独使用输出流返回一样,需要在返回信息前自己添加返回头,不然成功不了。另一个就是反射,要根据包名,加类名。

 File f = new File(this.getClass().getResource("/").getPath());

这个表示获取编译好的文件地址。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值