Java写一个简单的Web服务器Socket实现

一、实现思路

1、使用 ServerSocket 监听某一端口,然后等待连接获取 Socket对象。

2、创建一个类 HttpServer 继承 java.lang.Thread 类,重写 run()方法,执行浏览器请求。

3、获得浏览器请求,解析资源文件路径。

4、读取资源文件,响应给浏览器。

 

二、代码实现

1、ServerSocket 监听端口,获取 Socket对象

package com.httpserver.two;
import java.io.IOException;
import java.net.ServerSocket;
import java.net.Socket;
/**
 * @description: 使用socket 自己写一个web服务器 ---监听端口,获取socket 对象。
 * @version:v1.0
 * @author:w
 * @date:2018年6月6日上午11:03:36
 *
 */
public class WebServer {
	public void startServer(int port){
		try {
			@SuppressWarnings("resource")
			ServerSocket serverSocket = new ServerSocket(port);
			while(true){
				Socket socket = serverSocket.accept();
				new HttpServer(socket).start();
			}
		} catch (IOException e) {
			e.printStackTrace();
		}
	}
}

 

 

2、HttpServer.java类 --- 具体作用看方法注释。

package com.httpserver.two;

import java.io.BufferedReader;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.net.Socket;

/**
 * @description: 使用socket 实现 web服务器 --- 具体执行读写操作的。
 * @version:v1.0
 * @author:w
 * @date:2018年6月6日上午11:06:20
 *
 */
public class HttpServer extends Thread {
	/**
	 * web资源根路径
	 */
	public static final String ROOT = "c:/";
	
	/**
	 * 输入流对象,读取浏览器请求
	 */
	private InputStream input;
	
	/**
	 * 输出流对象,响应内容给浏览器
	 */
	private OutputStream out;

	/**
	 * @description:初始化socket对象,获取对应 输入,输出流
	 * @param socket
	 */
	public HttpServer(Socket socket) {
		try {
			input = socket.getInputStream();
			out = socket.getOutputStream();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

	/**
	 * 多线程方法调用
	 */
	@Override
	public void run() {
		String filePath = read();
		response(filePath);
	}

	/**
	 * @description: 读取资源文件,响应给浏览器。
	 * @param:@param filePath
	 *                   资源文件路径
	 * @return:void
	 * @version:v1.0
	 * @author:w
	 * @date:2018年6月6日 上午11:42:37
	 *
	 */
	private void response(String filePath) {
		File file = new File(ROOT + filePath);
		if (file.exists()) {
			// 1、资源存在,读取资源
			try {
				BufferedReader reader = new BufferedReader(new FileReader(file));
				StringBuffer sb = new StringBuffer();
				String line = null;
				while ((line = reader.readLine()) != null) {
					sb.append(line).append("\r\n");
				}
				StringBuffer result = new StringBuffer();
				result.append("HTTP /1.1 200 ok /r/n");
				result.append("Content-Type:text/html /r/n");
				result.append("Content-Length:" + file.length() + "/r/n");
				result.append("\r\n:" + sb.toString());
				out.write(result.toString().getBytes());
				out.flush();
				out.close();
			} catch (Exception e) {
				e.printStackTrace();
			}

		} else {
			// 2、资源不存在,提示 file not found
			StringBuffer error = new StringBuffer();
			error.append("HTTP /1.1 400 file not found /r/n");
			error.append("Content-Type:text/html \r\n");
			error.append("Content-Length:20 \r\n").append("\r\n");
			error.append("<h1 >File Not Found..</h1>");
			try {
				out.write(error.toString().getBytes());
				out.flush();
				out.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}

	}

	/**
	 * 
	 * @description:解析资源文件路径
	 * @example: GET /index.html HTTP/1.1
	 * @param:@return
	 * @return:String
	 * @version:v1.0
	 * @author:w
	 * @date:2018年6月6日 上午11:39:42
	 *
	 */
	private String read() {
		BufferedReader reader = new BufferedReader(new InputStreamReader(input));
		try {
			// 读取请求头, 如:GET /index.html HTTP/1.1
			String readLine = reader.readLine();
			String[] split = readLine.split(" ");
			if (split.length != 3) {
				return null;
			}
			System.out.println(readLine);
			return split[1];
		} catch (IOException e) {
			e.printStackTrace();
		}
		return null;
	}

}

三、代码测试

1、启动服务器

public static void main(String[] args) {
    new WebServer().startServer(8000);
}

 

2、浏览器地址栏输入: http://localhost:8000/index.html

 

3、效果如下:

 

    

java nio服务器源码 java写服务器_java nio服务器源码

 

四、总结

1、该示例代码可直接粘贴IDE中运行,无需任何第三方jar包。 简单简洁,便于理解。

2、c盘中,必须有一个 index.html的文件,若存放在其他位置,请修改 HttpServer.ROOT字段,路径位置即可。

3、若出现请求成功,响应页面为空白,请去掉 index.html 页面的 <html>标签试试。--- 至于什么问题导致的,目前还不清楚。 -_-!

4、 index.html 代码参考

            

<!DOCTYPE html>
<head>
<meta charset="UTF-8">
<title>Insert title here</title>
</head>
<body>
	<h1>Hello World! this is my webserver!</h1>
</body>

 

五、遗留问题解决

1、若出现请求成功,响应页面为空白,请去掉 index.html 页面的 <html>标签试试。原因是:换行符"\r\n"写错了,写成了"/r/n" 。

2、Firefox无法解析。原因是:"HTTP /1.1 200 ok" 中,"HTTP"和"/1.1" 直接不能有空格,否则 Firefox无法解析 。正确的是: "HTTP/1.1 200 ok"。

3、关于换行符问题补充: 

a. windows 换行符: "\r\n

b. linux 换行符: "\n

c. mac 换行符:"\r"

           4、可能引发问题: 假设在 linux 中,使用 windows的换行符会有效果吗?会引发上述的问题吗?

5、造成上述问题的发生,主要是 response header 不对导致的,chrome 浏览器可查看 response header 如下图:

java nio服务器源码 java写服务器_java nio服务器源码_02

 

六、补上修改后的代码 

package com.httpserver.two.fix;
import java.io.BufferedReader;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.OutputStream;
import java.io.OutputStreamWriter;
import java.net.Socket;
/**
 * @description: 使用socket 实现 web服务器 --- 具体执行读写操作的。<br>
 * 	fix: 1、换行符写错误:"/r/n" 调整为正确: "\r\n"。 <br>
 * 		 2、"HTTP /1.1 200 ok" 中,"HTTP"和"/1.1" 直接不能有空格,否则 Firefox无法解析。<br>
 * @version:v1.1
 * @author:w
 * @date:2018年12月18日 16:50:28
 */
public class HttpServerFix extends Thread {
	/**
	 * web资源根路径
	 */
	public static final String ROOT = "c:/";
	
	/**
	 * 输入流对象,读取浏览器请求
	 */
	private InputStream input;
	
	/**
	 * 输出流对象,响应内容给浏览器
	 */
	private OutputStream out;

	/**
	 * @description:初始化socket对象,获取对应 输入,输出流
	 * @param socket
	 */
	public HttpServerFix(Socket socket) {
		try {
			input = socket.getInputStream();
			out = socket.getOutputStream();
		} catch (IOException e) {
			e.printStackTrace();
		}
	}

	/**
	 * 多线程方法调用
	 */
	@Override
	public void run() {
		String filePath = read();
		response(filePath);
	}

	/**
	 * @description: 读取资源文件,响应给浏览器。
	 * @param:@param filePath
	 *                   资源文件路径
	 * @return:void
	 * @version:v1.0
	 * @author:w
	 * @date:2018年6月6日 上午11:42:37
	 *
	 */
	private void response(String filePath) {
		File file = new File(ROOT + filePath);
		if (file.exists()) {
			// 1、资源存在,读取资源
			try {
				BufferedReader reader = new BufferedReader(new FileReader(file));
				StringBuffer sb = new StringBuffer();
				String line = null;
				while ((line = reader.readLine()) != null) {
					System.out.println("line:"+ line);
					sb.append(line).append("\r\n");
				}
				StringBuffer result = new StringBuffer();
				/**
				 * 1、 换行符"/r/n"写错了,正确的是: "\r\n"
				 * 2、"HTTP /1.1 ..."之间不能有空格,否则Firfox不能解析
				 * 3、 关于换行符的补充:
				 * 	  a.windows:"\r\n"
				 *    b.linux:"\n"
				 *    c.mac:"\r" 
				 */
			//	result.append("HTTP /1.1 200 ok /r/n");
				result.append("HTTP/1.1 200 ok \r\n");
				result.append("Content-Language:zh-CN \r\n");
				// charset=UTF-8 解决中文乱码问题
				result.append("Content-Type:text/html;charset=UTF-8 \r\n");
				result.append("Content-Length:" + file.length() + "\r\n");
				result.append("\r\n" + sb.toString());
				out.write(result.toString().getBytes());
				out.flush();
				out.close();
			} catch (Exception e) {
				e.printStackTrace();
			}

		} else {
			// 2、资源不存在,提示 file not found
			StringBuffer error = new StringBuffer();
			error.append("HTTP/1.1 400 file not found \r\n");
			error.append("Content-Type:text/html \r\n");
			error.append("Content-Length:20 \r\n").append("\r\n");
			error.append("<h1 >File Not Found..</h1>");
			try {
				out.write(error.toString().getBytes());
				out.flush();
				out.close();
			} catch (IOException e) {
				e.printStackTrace();
			}
		}

	}

	/**
	 * 
	 * @description:解析资源文件路径
	 * @example: GET /index.html HTTP/1.1
	 * @param:@return
	 * @return:String
	 * @version:v1.0
	 * @author:w
	 * @date:2018年6月6日 上午11:39:42
	 *
	 */
	private String read() {
		BufferedReader reader = new BufferedReader(new InputStreamReader(input));
		try {
			// 读取请求头, 如:GET /index.html HTTP/1.1
			String readLine = reader.readLine();
			String[] split = readLine.split(" ");
			if (split.length != 3) {
				return null;
			}
			System.out.println(readLine);
			return split[1];
		} catch (IOException e) {
			e.printStackTrace();
		}
		return null;
	}

}