tomcat学习笔记(一) 模拟一个简单的web server

        之前看视频的过程中看到一个讲tomcat的视频,从那之后才知道自己对tomcat可以说是一无所知。我所说的一无所知意思是对tomcat的工作原理一无所知,当然使用tomcat部署项目肯定是没问题的。我相信肯定有很多朋友和我一样,虽然每天用tomcat,但是对tomcat基本不了解。因此我决定学一学tomcat,网上纯粹讲tomcat的视频基本上没有,昨晚偶然看到一本书,叫做《how tmocat works》,这本书正是我想要的。现在我一边看书,一边写下自己的学习过程,也希望能够帮到没时间看那些冗长的书的朋友们。废话不说了。

        一、HTTP协议。

        协议的意思其实就是一个约定,比如A和B约定以后把某个餐馆叫做C,以后A和B说咱们去C吃饭吧,B就知道A说的是去那个餐馆吃饭,这就是个协议。HTTP协议就是类似这个例子的,只不过它的协议双方是客户端和服务器,客户端发送一个请求,有了这个协议后,服务器就知道这个请求的每个部分代表什么意思,这个客户端和服务器才能进行交流,否则服务器不知道客户端在说什么,客户端不知道服务器在说什么,就没办法交流了。

        简单来说访问网站的过程是这个:用户在浏览器输入一个地址,摁回车之后,就会向服务器发送一个请求,告诉服务器它想要一个页面,服务器找到这个页面,然后给浏览器发送回来。

       1、请求。当我们在页面上点击一个提交按钮的时候,会向服务器发送一个请求,对这个请求,我的理解是它就是个IO流(如果不理解也可以理解为字符串),就是向服务器发送了个字符串,这个字符串里面包含了一些信息,关于这个,我可能理解得不太对,但是本质应该就是这样的。这个字符串的格式如下:

       POST /examples/default.jsp HTTP/1.0

       Accept: text/plain;text/html

       Accept-Language: en-gb

       Connection:Keep-Alive

       Host: localhost

       User-Agent: Mozilla/4.0(compatible;MSIE:4.01;Widows 98)

       Content-Length: 33

       Content-Type:application/x-www-form-urlencoded

       Accept-Encodding:gzip,deflate


      lastName=Franks&firstName=Michael

      相信大家能够看懂上面字符串的意思,第一行是请求方法 空格 请求资源路径 协议/协议版本,中间那部分是请求头,最后一行是请求体,这里用的POST方法,所以表单的参数会在请求体中,这行和上面的用一个空行隔开,以便解析此请求的时候能知道往下就是请求体了。

     2、响应。服务器接收到用户的请求之后,会找到用户所要的资源(页面,图片等),然后形成一个字符串,再发送到客户端。响应的格式如下:

    HTTP/1.1 200 OK
   Server: Microsoft-IIS/4.0
   Date: Mon, 5 Jan 2004 13:13:33 GMT
   Content-Type: text/html
   Last-Modified: Mon, 5 Jan 2004 13:13:12 GMT
   Content-Length: 112


   <html>
   <head>
   <title>HTTP Response Example</title>
   </head>
   <body>
   Welcome to Brainy Software
   </body>
   </html>

    第一行的意思:协议/协议版本 空格 状态码(成功,失败等状态用数字来表示) 空格 状态描述(状态码人们看不懂,所以这里有状态描述),接下来是告诉客户端的一些信息,再接下来就是客户请求的页面的内容。

   3、写这么详细是担心有些刚入门的新手不太理解,但是可能在很多人看来都是没用的,因为自己都知道了,所以我还是大概写一写,如果大家有不理解的地方可以提问。

   4、其实web server的原理也很简单,用的东西都是咱们会的。在服务器上运行一个ServerSocket程序,这个程序可以监控端口,底层的东西说我也不清楚。但是用过socket的人都知道它可以监控,我们在浏览器发送一个请求的时候,请求会被传送到服务器上,如果我们用的端口是这个程序所监控的端口,那么这个程序就会接收到这个请求,因此请求里面包含了所需要的页面,因此程序会解析请求(因为请求就是IO流,所以可以用JAVA中IO的操作来解析),解析后会找到请求所需要的页面,页面其实就是个文件,所以用JAVA的文件操作,读出这个页面,然后通过SOCKET就可以写到客户端了,和我们之前做的文件读写是一样,只不过这里是用SOCKET写到远程了。

   5、接下来就看代码吧。以下的代码是我照着《how tocat works》中的打出来的,已经运行通过,大家可以参考。

    需要建三个类:HttpServer、Request、Response。

   

package com.tomcat;

import java.io.File;
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 HttpServer {
	public static final String  WEB_ROOT=System.getProperty("user.dir") + File.separatorChar + "webRoot";
	private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";
	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(Exception e){
			e.printStackTrace();
			System.exit(1);
		}
		while(!shutdown){
			Socket socket = null;
			InputStream in = null;
			OutputStream out = null;
			try {
				socket = serverSocket.accept();
				in = socket.getInputStream();
				out = socket.getOutputStream();
				Request request = new Request(in);
				request.parse();
				Response response = new Response(out);
				response.setRequest(request);
				response.sendStaticResource();
				socket.close();
				shutdown = request.getUri().equals("SHUTDOWN_COMMAND");
			} catch (IOException e) {
				e.printStackTrace();
				continue;
			}
		}
	}
}

package com.tomcat;

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

public class Request {
	private InputStream input;
	private String uri;
	public Request(InputStream input){
		this.input = input;
	}
	public void parse(){
		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.println(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;
	}
}


package com.tomcat;

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

public class Response {
	private static final int BUFFER_SIZE=1024;
	Request request;
	OutputStream out;
	public Response(OutputStream out){
		this.out = out;
	}
	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){
					out.write(bytes,0,ch);
					ch = fis.read(bytes,0,BUFFER_SIZE);
				}
			}else{
				String errorMessage = "HTTP/1.1 404 File Not Found\n" + 
					"Content-Type:text/html\n" +
					"Content-Length:23\n" +
					"\n" + 
					"<h1>File not Found</h1>";
				out.write(errorMessage.getBytes());
			} 
		}catch(Exception e){
			System.out.println(e.toString());
		}finally{
			if(fis != null)
				fis.close();
		}
	}
}

    相信大家都能看懂这些代码,其实很简单,就是Socket和IO读写。

    建一个webroot目录,放入一个index.html,这个html代码可以随便写点。

    将以上代码编译完成后,运行HttpServer类,然后在浏览器输入http://localhost:8080/index.html,即可看到index.html中的内容。

    此文章是讲的web server的原理,因此是服务于静态资源的,访问JSP等动态资源的会在以后的章节中讲到。

    希望大家能够指出我的错误,如果有问题非常欢迎大家和我讨论,共同学习。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章