java實現(xiàn)一個簡單的Web服務(wù)器實例解析
Web服務(wù)器也稱為超文本傳輸協(xié)議服務(wù)器,使用http與其客戶端進(jìn)行通信,基于java的web服務(wù)器會使用兩個重要的類,
java.net.Socket類和java.net.ServerSocket類,并基于發(fā)送http消息進(jìn)行通信。
這個簡單的Web服務(wù)器會有以下三個類:
*HttpServer
*Request
*Response
應(yīng)用程序的入口在HttpServer類中,main()方法創(chuàng)建一個HttpServer實例,然后調(diào)用其await()方法,顧名思義,await()方法會在指定端口上等待HTTP請求,對其進(jìn)行處理,然后發(fā)送響應(yīng)信息回客戶端,在接收到關(guān)閉命令前,它會保持等待狀態(tài)。
該應(yīng)用程序僅發(fā)送位于指定目錄的靜態(tài)資源的請求,如html文件和圖像,它也可以將傳入到的http請求字節(jié)流顯示到控制臺,但是,它并不發(fā)送任何頭信息到瀏覽器,如日期或者cookies等。
下面為這幾個類的源碼
Request:
package cn.com.server;
import java.io.InputStream;
public class Request {
private InputStream input;
private String uri;
public Request(InputStream input){
this.input=input;
}
public void parse(){
//Read a set of characters from the socket
StringBuffer request=new StringBuffer(2048);
int i;
byte[] buffer=new byte[2048];
try {
i=input.read(buffer);
}
catch (Exception e) {
e.printStackTrace();
i=-1;
}
for (int j=0;j<i;j++){
request.append((char)buffer[j]);
}
System.out.print(request.toString());
uri=parseUri(request.toString());
}
public 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 this.uri;
}
}
Request類表示一個HTTP請求,可以傳遞InputStream對象來創(chuàng)建Request對象,可以調(diào)用InputStream對象中的read()方法來讀取HTTP請求的原始數(shù)據(jù)。
上述源碼中的parse()方法用于解析Http請求的原始數(shù)據(jù),parse()方法會調(diào)用私有方法parseUrI()來解析HTTP請求的URI,除此之外,并沒有做太多的工作,parseUri()方法將URI存儲在變量uri中,調(diào)用公共方法getUri()會返回請求的uri。
Response:
package cn.com.server;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
/**
* HTTP Response = Status-Line
* *(( general-header | response-header | entity-header ) CRLF)
* CRLF
* [message-body]
* Status-Line=Http-Version SP Status-Code SP Reason-Phrase CRLF
*
*/
public class Response {
private static final int BUFFER_SIZE=1024;
Request request;
OutputStream output;
public Response(OutputStream output){
this.output=output;
}
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){
output.write(bytes, 0, BUFFER_SIZE);
ch=fis.read(bytes, 0, BUFFER_SIZE);
}
} else{
//file not found
String errorMessage="HTTP/1.1 404 File Not Found\r\n"+
"Content-Type:text/html\r\n"+
"Content-Length:23\r\n"+
"\r\n"+
"<h1>File Not Found</h1>";
output.write(errorMessage.getBytes());
}
}
catch (Exception e) {
System.out.println(e.toString());
}
finally{
if(fis!=null){
fis.close();
}
}
}
}
Response對象在HttpServer類的await()方法中通過傳入套接字中獲取的OutputStream來創(chuàng)建。
Response類有兩個公共方法:setRequest()和sendStaticResource() ,setRequest()方法會接收一個Request對象為參數(shù),sendStaticResource()方法用于發(fā)送一個靜態(tài)資源到瀏覽器,如Html文件。
HttpServer:
package cn.com.server;
import java.io.File;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.InetAddress;
import java.net.ServerSocket;
import java.net.Socket;
public class HttpServer {
/**
* WEB_ROOT is the directory where our html and other files reside.
* For this package,WEB_ROOT is the "webroot" directory under the
* working directory.
* the working directory is the location in the file system
* from where the java command was invoke.
*/
public static final String WEB_ROOT=System.getProperty("user.dir")+File.separator+"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(0);
}
while(!shutdown){
Socket socket=null;
InputStream input=null;
OutputStream output=null;
try {
socket=serverSocket.accept();
input=socket.getInputStream();
output=socket.getOutputStream();
//create Request object and parse
Request request=new Request(input);
request.parse();
//create Response object
Response response=new Response(output);
response.setRequest(request);
response.sendStaticResource();
}
catch (Exception e) {
e.printStackTrace();
continue;
}
}
}
}
這個類表示一個Web服務(wù)器,這個Web服務(wù)器可以處理對指定目錄的靜態(tài)資源的請求,該目錄包括由公有靜態(tài)變量final WEB_ROOT指明的目錄及其所有子目錄。
現(xiàn)在在webroot中創(chuàng)建一個html頁面,命名為index.html,源碼如下:
<!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Insert title here</title> </head> <body> <h1>Hello World!</h1> </body> </html>
現(xiàn)在啟動該WEB服務(wù)器,并請求index.html靜態(tài)頁面。

所對應(yīng)的控制臺的輸出:

如此,一個簡單的http服務(wù)器便完成了。
總結(jié)
以上就是本文關(guān)于java實現(xiàn)一個簡單的Web服務(wù)器實例解析的全部內(nèi)容,希望對大家有所幫助。感興趣的朋友可以繼續(xù)參閱本站其他相關(guān)專題,如有不足之處,歡迎留言指出。感謝朋友們對本站的支持!
相關(guān)文章
Java Swing組件布局管理器之FlowLayout(流式布局)入門教程
這篇文章主要介紹了Java Swing組件布局管理器之FlowLayout(流式布局),結(jié)合實例形式分析了Swing組件布局管理器FlowLayout流式布局的常用方法及相關(guān)使用技巧,需要的朋友可以參考下2017-11-11
Spring Boot Web 靜態(tài)文件緩存處理的方法
本篇文章主要介紹了Spring Boot Web 靜態(tài)文件緩存處理的方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2018-02-02
Java中循環(huán)冗余校驗(CRC32)的實現(xiàn)
CRC校驗實用程序庫在數(shù)據(jù)存儲和數(shù)據(jù)通訊領(lǐng)域,為了保證數(shù)據(jù)的正確,就不得不采用檢錯的手段,下面這篇文章主要給大家介紹了關(guān)于Java中循環(huán)冗余校驗(CRC32)實現(xiàn)的相關(guān)資料,需要的朋友可以參考借鑒,下面來一起看看吧。2017-10-10
淺談s:select 標(biāo)簽中l(wèi)ist存放map對象的使用
下面小編就為大家?guī)硪黄獪\談s:select 標(biāo)簽中l(wèi)ist存放map對象的使用。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-11-11
MyBatis-Plus+達(dá)夢數(shù)據(jù)庫實現(xiàn)高效數(shù)據(jù)持久化的示例
這篇文章主要介紹了MyBatis-Plus和達(dá)夢數(shù)據(jù)庫實現(xiàn)高效數(shù)據(jù)持久化,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-08-08

