簡(jiǎn)單實(shí)現(xiàn)Java web服務(wù)器
一個(gè)簡(jiǎn)單的Java web服務(wù)器實(shí)現(xiàn),比較簡(jiǎn)單,基于java.net.Socket和java.net.ServerSocket實(shí)現(xiàn);
一、程序執(zhí)行步驟
1.創(chuàng)建一個(gè)ServerSocket對(duì)象;
2.調(diào)用ServerSocket對(duì)象的accept方法,等待連接,連接成功會(huì)返回一個(gè)Socket對(duì)象,否則一直阻塞等待;
3.從Socket對(duì)象中獲取InputStream和OutputStream字節(jié)流,這兩個(gè)流分別對(duì)應(yīng)request請(qǐng)求和response響應(yīng);
4.處理請(qǐng)求:讀取InputStream字節(jié)流信息,轉(zhuǎn)成字符串形式,并解析,這里的解析比較簡(jiǎn)單,僅僅獲取uri(統(tǒng)一資源標(biāo)識(shí)符)信息;
5.處理響應(yīng):根據(jù)解析出來(lái)的uri信息,從WEB_ROOT目錄中尋找請(qǐng)求的資源資源文件, 讀取資源文件,并將其寫入到OutputStream字節(jié)流中;
6.關(guān)閉Socket對(duì)象;
7.轉(zhuǎn)到步驟2,繼續(xù)等待連接請(qǐng)求;
二、代碼實(shí)現(xiàn)
服務(wù)器實(shí)現(xiàn):
package ex01.pyrmont;
import java.net.Socket;
import java.net.ServerSocket;
import java.net.InetAddress;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.IOException;
import java.io.File;
public class HttpServer {
/**
* WEB_ROOT是HTML和其它文件存放的目錄. 這里的WEB_ROOT為工作目錄下的webroot目錄
*/
public static final String WEB_ROOT = System.getProperty("user.dir") + File.separator + "webroot";
// 關(guān)閉服務(wù)命令
private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";
public static void main(String[] args) {
HttpServer server = new HttpServer();
//等待連接請(qǐng)求
server.await();
}
public void await() {
ServerSocket serverSocket = null;
int port = 8080;
try {
//服務(wù)器套接字對(duì)象
serverSocket = new ServerSocket(port, 1, InetAddress.getByName("127.0.0.1"));
} catch (IOException e) {
e.printStackTrace();
System.exit(1);
}
// 循環(huán)等待一個(gè)請(qǐng)求
while (true) {
Socket socket = null;
InputStream input = null;
OutputStream output = null;
try {
//等待連接,連接成功后,返回一個(gè)Socket對(duì)象
socket = serverSocket.accept();
input = socket.getInputStream();
output = socket.getOutputStream();
// 創(chuàng)建Request對(duì)象并解析
Request request = new Request(input);
request.parse();
// 檢查是否是關(guān)閉服務(wù)命令
if (request.getUri().equals(SHUTDOWN_COMMAND)) {
break;
}
// 創(chuàng)建 Response 對(duì)象
Response response = new Response(output);
response.setRequest(request);
response.sendStaticResource();
// 關(guān)閉 socket 對(duì)象
socket.close();
} catch (Exception e) {
e.printStackTrace();
continue;
}
}
}
}
Request類:
package ex01.pyrmont;
import java.io.InputStream;
import java.io.IOException;
public class Request {
private InputStream input;
private String uri;
public Request(InputStream input) {
this.input = input;
}
//從InputStream中讀取request信息,并從request中獲取uri值
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.print(request.toString());
uri = parseUri(request.toString());
}
/**
*
* requestString形式如下:
* GET /index.html HTTP/1.1
* Host: localhost:8080
* Connection: keep-alive
* Cache-Control: max-age=0
* ...
* 該函數(shù)目的就是為了獲取/index.html字符串
*/
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;
}
}
Response類:
package ex01.pyrmont;
import java.io.OutputStream;
import java.io.IOException;
import java.io.FileInputStream;
import java.io.File;
/*
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 {
//將web文件寫入到OutputStream字節(jié)流中
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, ch);
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) {
// thrown if cannot instantiate a File object
System.out.println(e.toString());
} finally {
if (fis != null)
fis.close();
}
}
}
三、結(jié)果測(cè)試
訪問(wèn)存在的資源文件(注意存放在工程目錄的webroot文件夾里):

訪問(wèn)不存在的資源文件:

關(guān)閉服務(wù)器:

參考資料:《深入剖析Tomcat》
@author 風(fēng)一樣的碼農(nóng)
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Springboot+Vue+axios實(shí)現(xiàn)文章收藏功能
這篇文章主要為大家詳細(xì)介紹了Springboot+Vue+axios實(shí)現(xiàn)文章收藏功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-08-08
Java實(shí)現(xiàn)一個(gè)簡(jiǎn)易聊天室流程
這篇文章主要介紹了我的java課程設(shè)計(jì)一個(gè)多人聊天室(socket+多線程)本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-11-11
SpringBoot集成Mybatis實(shí)現(xiàn)對(duì)多數(shù)據(jù)源訪問(wèn)原理
本文主要分析討論在SpringBoot應(yīng)用中我們?cè)撊绾闻渲肧qlSessionFactoryBean對(duì)象,進(jìn)而實(shí)現(xiàn)對(duì)多個(gè)不同的數(shù)據(jù)源的操縱,文章通過(guò)代碼示例介紹的非常詳細(xì),需要的朋友可以參考下2023-11-11
Spring中的@ConfigurationProperties在方法上的使用詳解
這篇文章主要介紹了Spring中的@ConfigurationProperties在方法上的使用詳解,@ConfigurationProperties應(yīng)該經(jīng)常被使用到,作用在類上的時(shí)候,將該類的屬性取值?與配置文件綁定,并生成配置bean對(duì)象,放入spring容器中,提供給其他地方使用,需要的朋友可以參考下2024-01-01
spring注解如何為bean指定InitMethod和DestroyMethod
這篇文章主要介紹了spring注解如何為bean指定InitMethod和DestroyMethod,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-11-11
SpringBoot自定義注解及AOP的開(kāi)發(fā)和使用詳解
在公司項(xiàng)目中,如果需要做一些公共的功能,如日志等,最好的方式是使用自定義注解,自定義注解可以實(shí)現(xiàn)我們對(duì)想要添加日志的方法上添加,這篇文章基于日志功能來(lái)講講自定義注解應(yīng)該如何開(kāi)發(fā)和使用,需要的朋友可以參考下2023-08-08

