这是我的第一篇文章。和众多程序员一样,非常希望去分享自己的知识。但是万事开头难,也可能是我自己水平的问题,真是想不到要从什么内容开始。于是趁着有时间,就自己捣鼓了一下HTTP服务器。后来想想干脆就以自己手写一个简单的HTTP服务器的教程来开始我自己的写作吧。本文中不足和错误的地方请各位看客多多指教。下面就开始吧。
HTTP简介
当用户在浏览器中输入一个指向特定网页的URL地址时,浏览器就会生成一个HTTP请求,因为浏览器需要将HTTP请求发送的服务器,所以这时浏览器会与服务器建立TCP连接,当TCP可靠连接建立之后,浏览器会将生成的HTTP请求发送到服务器端。这时服务器程序接收到了信息将要去识别这个信息的内容。服务器程序识别之后是一个HTTP的请求。就调用相应的服务程序,经过服务程序的分析和处理之后服务器端知道需要返回什么内容给浏览器了。当服务器返回了内容给浏览器后,这时浏览器与服务器之间的数据交换完毕,这时TCP可靠连接就会断开。如果用户希望访问新的网页,浏览器就必须再次建立与服务器的连接了。
开始之前我们准备好工具
- java环境(JDK1.7)
- eclipse
- chrome浏览器
开始
使用eclipse建立项目结构如图
HTTPServer.java为程序代码
abc.html为需要请求的html页面
HTTPServer.java 代码
package server;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.ServerSocket;
import java.net.Socket;
public class HTTPServer {
public static void main(String[] args) {
int port;
ServerSocket serverSocket;
try{
port=Integer.parseInt(args[0]);
}catch(Exception e){
System.out.println("port=8080(默认)");
port=8080;
}
try {
serverSocket=new ServerSocket(port);
System.out.println("服务器正在监听端口:"+serverSocket.getLocalPort());
while(true){
Socket socket=serverSocket.accept(); //没有时会阻塞 程序停在此处
System.out.println("建立了与客户的一个新的TCP连接,该客户的地址为:"+socket.getInetAddress()+":"+socket.getPort());
try {
service(socket);
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
public static void service(Socket socket) throws Exception{
InputStream socketIn=socket.getInputStream();
Thread.sleep(500);
int size=socketIn.available();
byte[] buffer=new byte[size];
socketIn.read(buffer);
String request=new String(buffer);
System.out.println(request);
//获取HTTP请求的第一行
String firstLineOfRequest=request.substring(0,request.indexOf("\r\n"));
System.out.println("http请求的第一行:"+firstLineOfRequest);
//解析HTTP请求的第一行
String[] parts=firstLineOfRequest.split(" ");
//获取HTTP请求中的URI
String uri=parts[1];
/*决定HTTP响应正文的类型*/
String contentType;
if(uri.indexOf("html")!=-1||uri.indexOf("htm")!=-1)
contentType="text/html";
else if(uri.indexOf("jpg")!=-1||uri.indexOf("jpeg")!=-1)
contentType="image/jpeg";
else if(uri.indexOf("gif")!=-1)
contentType="image/gif";
else
contentType="application/octet-stream"; //字节流类型
/*创建HTTP响应结果*/
//HTTP响应的第一行
String responseFirstLine="HTTP/1.1 200 OK\r\n";
String responseHeader="Content-type:"+contentType+"\r\n\r\n";
InputStream in=HTTPServer.class.getResourceAsStream("/server"+uri);
/*发送HTTP响应结果*/
OutputStream socketOut=socket.getOutputStream();
socketOut.write(responseFirstLine.getBytes());
socketOut.write(responseHeader.getBytes());
//发送正文
int len=0;
buffer=new byte[128];
while((len=in.read(buffer))!=-1)
socketOut.write(buffer,0,len);
Thread.sleep(1000);
socket.close();
}
}
abc.html
this Html
程序默认监听端口8080
在一般情况下accept()函数是阻塞式的,当没有socket连接的时候本线程是停在此处的。
当有一个TCP请求建立连接时这时服务器程序的accept()返回一个Socket对象。这个Socket对象就是在本程序中充当客户端的角色对象。可以从此个对象中获取输入流读取信息进来,也可以通过输出流返回信息给客户端。
这里我们通过浏览器进行测试
将这个程序跑起来
我们使用浏览器进行请求http://localhost:8080/abc.html 请求自己服务器的abc.html
查看控制台打印出来的内容
port=8080(默认)
服务器正在监听端口:8080
建立了与客户的一个新的TCP连接,该客户的地址为:/0:0:0:0:0:0:0:1:52085
GET /abc.html HTTP/1.1
Host: localhost:8080
Connection: keep-alive
Cache-Control: max-age=0
Upgrade-Insecure-Requests: 1
User-Agent: Mozilla/5.0 (Windows NT 10.0; WOW64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/51.0.2704.103 Safari/537.36
Accept: text/html,application/xhtml+xml,application/xml;q=0.9,image/webp,*/*;q=0.8
Accept-Encoding: gzip, deflate, sdch
Accept-Language: zh-CN,zh;q=0.8
http请求的第一行:GET /abc.html HTTP/1.1
从控制台打印结果中我们可以看到HTTP协议的大致结构
- 请求行
- 消息报头
- 请求正文
请求行中包括了本次HTTP协议是什么方式,以及uri定位地址在哪里 和一些关于HTTP规范版本的信息。
结语
文章内容比较简单基础,但这是关于底层的基础内容。我想这些机理还是很重要的,需要牢记于心,以便于掌控。这样才能在碰到问题的时候迅速解决问题。
哈哈
谢谢大家阅读!