1.概括
熟悉tomcat的都知道,Catalina 中有两个主要的模块:连接器和容器。本文将写一个可以创建更好的请求和响应对象的连接器,改进系列2和系列3实现的功能。根据Servlet 2.3 和 2.4规范的连 接 器 必 须 创 建 javax.servlet.http.HttpServletRequest 和javax.servlet.http.HttpServletResponse,并传递给被调用的 servlet 的 service 方法。
从本文开始,手写tomcat的核心骨架将逐步呈现出来,本文将展示三个核心模块:分别是connector、startup和core。
各模块分工概要如下:
1.startup模块在本文只有一个类Bootstrap,是启动tomcat server的入口。
2.core模块作用是处理请求,本文中由两个类组成,分别为ServletProcessor和StaticResourceProcessor,读过本系列上面文章的童鞋应该对这两个类不陌生,后面的系列这两个类将逐步有各种容器替代。
3.connector 模块,即本文的重点,该模块的类可以分为五组:
(1) 连接器和它的支撑类(HttpConnector 和 HttpProcessor)。
(2) HTTP 请求的类(HttpRequest)和它的辅助类。
(3) HTTP 响应的类(HttpResponse)和它的辅助类。
(4) Facade 类(HttpRequestFacade 和 HttpResponseFacade)。
(5) Constant 类
UML图如下(StringManager类用于处理错误信息,是辅助类,可忽略,不影响理解整体功能。):
2.startup实现
public final class Bootstrap {
public static void main(String[] args) {
HttpConnector connector = new HttpConnector();
connector.start();
}
}
Boot
代码很简单,就是创建connector实例并启动connector,开始工作,下面代码将从HttpConnector类展开。
3.connector实现
HttpConnector类实现
import java.io.IOException;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
public class HttpConnector implements Runnable {
boolean stopped;
private String scheme = "http";
public String getScheme() {
return scheme;
}
public void run() {
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 (!stopped) {
// Accept the next incoming connection from the server socket
Socket socket = null;
try {
socket = serverSocket.accept();
}
catch (Exception e) {
continue;
}
// Hand this socket off to an HttpProcessor
HttpProcessor processor = new HttpProcessor(this);
processor.process(socket);
}
}
public void start() {
Thread thread = new Thread(this);
thread.start();
}
}
代码很简单,归纳几点:
1.HttpConnector implements Runnable,由Bootstrap类调用HttpConnector的start方法启动连接器线程。
2.HttpConnector线程就做一件事情,收到一个请求就会创建一个HttpProcessor来处理请求,收到N个请求,就会创建N个HttpProcessor实例。
HttpProcessor类实现
进阶HttpConnector,我们来看一下如何处理请求。
(未完。。。。。)