前几天有个做python的老哥问我啥是tomcat,我从python的flask说到springMVC,又从MVC说到Tomcat,但是感觉还是没解释清,今天突然想起来以前听过一节公开课,是手动实现一个自己的Tomcat(自己都能实现一个Tomcat了,原理自然就懂了),虽然找不到上课视频了,但是找了半天在我的老电脑里找到了以前写的源码,分享一下,希望能对大家有一点帮助。
首先我们新建一个Java工程MyTomcat,什么包都不用导入,只用原生的jdk1.8即可(其他版本也OK),目录结构如下:
第一步我们实现主启动类Server.java,在这个类里创建一个ServerSocket,不断循环等待连接他的Socket,也就是不断的等待客户端(浏览器)访问,内容如下:
package com.sunsy.tomcat;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class Server {
private static ServerSocket serverSocket;
private static int port = 8080;
private final static int POOL_SIZE = 8;
private static ExecutorService executorService;
public static void start() {
try {
serverSocket = new ServerSocket(port);
Socket socket = null;
System.out.println("start server, port : " + port);
executorService = Executors.newFixedThreadPool(POOL_SIZE);
while (true) {
socket = serverSocket.accept();
executorService.execute(new Handler(socket));
}
} catch (IOException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
start();
}
}
当有socket连接该ServerSocket时,会向线程池里放一个任务,也就是我们接下来要写的Handler.java类,这个类构建时需要把连接到ServerSocket的socket传入,从这个socket的输入流里我们能看到请求的信息(比如启动我们自己的Server后,用浏览器访问127.0.0.1:8080,然后在socket的输入流里我们能看到“GET / HTTP/1.1”这样的信息),通过这些信息我们就能构建我们自己的Request了,然后通过这个Request里的路径信息以及web.xml里的信息,我们可以利用反射得到我们自己写的HttpServlet类,这个HttpServlet会向socket的输出流里写数据(也就是写到我们自己的Response里,也就是dispatcher方法),这些数据就能被浏览器获取到,代码如下:
package com.sunsy.tomcat;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.net.Socket;
import com.sunsy.servlet.HttpServlet;
public class Handler implements Runnable {
private Socket socket;
private PrintWriter writer;
public Handler(Socket socket) {
this.socket = socket;
}
@Override
public void run() {
try {
writer = new PrintWriter(new OutputStreamWriter(socket.getOutputStream(), "UTF-8"));
writer.println("HTTP/1.1 200 OK");
writer.println("Content-Type: text/html;charset=utf-8");
writer.println();
Request request = new Request();
Response response = new Response(writer);
InputStream inputStream = socket.getInputStream();
BufferedReader reader = new BufferedReader(new InputStreamReader(inputStream));
while(true) {
String msg = reader.readLine();
if(msg==null || "".equals(msg.trim())) {
break;