深入剖析tomcat之一個簡單的web伺服器

来源:http://www.cnblogs.com/archy_yu/archive/2016/04/08/5370244.html
-Advertisement-
Play Games

這個簡單的web伺服器包含三個類 HttpServer Request Response 在應用程式的入口點,也就是靜態main函數中,創建一個HttpServer實例,然後調用其await()方法。顧名思義,await方法會在制定的埠上等待http請求,並對其進行處理,然後發送相應的消息回客戶端 ...


這個簡單的web伺服器包含三個類

  • HttpServer
  • Request
  • Response

在應用程式的入口點,也就是靜態main函數中,創建一個HttpServer實例,然後調用其await()方法。顧名思義,await方法會在制定的埠上等待http請求,並對其進行處理,然後發送相應的消息回客戶端。在接收到命令之前,它會一直保持等待的狀態。

  • HttpServer類
package simpleHttpServer;

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.separator
            + "webroot";
    
    private static final String SHUTDOWN_COMMAND = "/SHUTDOWN";
    
    private boolean shudown = 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 (IOException e){
            e.printStackTrace();
            System.exit(1);
        }
        
        while(!this.shudown){
            Socket socket = null;
            InputStream input = null;
            OutputStream output = null;
            
            try{
                
                socket = serverSocket.accept();
                input = socket.getInputStream();
                output = socket.getOutputStream();
                
                Request request = new Request(input);
                request.parse();
                
                Response response = new Response(output);
                response.setRequest(request);
                response.sendStaticResource();
                
                socket.close();
                
                this.shudown = request.getUri().equals(SHUTDOWN_COMMAND);
                
            }
            catch (Exception e){
                e.printStackTrace();
                continue;
            }
        }
        
    }
        
}

這個簡單的web伺服器,可以處理指定目錄中的靜態資源請求;用WEB_ROOT表示制定的目錄

public static final String WEB_ROOT = System.getProperty("user.dir") + File.separator + "webroot";

這裡是指當前目錄下的webroot文件夾下麵的資源。
我們通過在游覽器中輸入這樣的內容,進行資源的請求:
http://127.0.0.1:8080/index.html

  • Request類
    Request類表示一個Http請求,可以傳遞InputStream對象來創建Request對象,調用InputStream對象的read進行Http請求數據的讀取。
package simpleHttpServer;

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 (Exception e){
            e.printStackTrace();
            i = -1;
        }
        
        for(int j=0;j<i;j++){
            request.append((char)buffer[j]);
        }
        
        System.out.print(request.toString());
        this.uri = this.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 this.uri;
    }
    
}

Request類最重要的兩個函數是parse和ParseUri;parse()方法會調用私有方法parseUri來解析HTTP請求的uri,初次之外,並沒有做太多的工作。parseuri會將解析的URI存儲在變數uri中。

我們以 http://127.0.0.1:8080/index.html 請求為例,HTTP請求的請求行為
GET /index.html HTTP/1.1
parse()方法從傳入的Request對象的InputStream對象中讀取整個位元組流,並且將位元組數組存入緩衝區。然後用緩存區的數組初始化StringBuffer對象request。 這樣再解析StringBuffer就可以解析到Uri。

*Response類
Response類表示Http相應。其定義如下

package simpleHttpServer;

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;
    private Request request;
    private 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, ch);
                    ch = fis.read(bytes, 0, BUFFER_SIZE);
                }
            }
            else{
                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){
            e.printStackTrace();
        }
        finally{
            if(fis != null){
                fis.close();
            }
        }
        
    }
    
    
}

使用OutputStream和Request來初始化Reponse,Response比較簡單,得到Request的Uri,然後讀取對應的file,如果file存在,則將file中的數據讀取到緩存中,並且發送給游覽器;如果file不存在,那麼就發送

"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>";

錯誤信息給游覽器。

我們在eclipse中將代碼跑起來,在游覽器中輸入
http://127.0.0.1:8080/index.html

http://127.0.0.1:8080/index.php

一個簡單的額web伺服器就跑起來了!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 一、源文件的編寫 Java是完全面向對象的語言,所以Java的所有操作都是基於類(class)完成的。Java中所有程式的代碼都需要放在一個類中,類用關鍵字class聲明,在class之前可以添加一些修飾符,Java應用程式的源文件由若幹個書寫形式相互獨立的類組成。 1.例子: (1)import語 ...
  • 本人在寫學生信息管理系統時遇到一個很頭疼的錯誤——error LNK2005重覆定義錯誤,苦思冥想百度谷歌bing之後都沒能解決問題,於一清早剎那間覺得知道問題出在哪兒了,於是乎起床、開機、修改代碼一氣呵成,終於0 error(s)\0 warning(s)。 error LNK2005錯誤分為好幾 ...
  • java線程的五種狀態其實要真正高清,只需要明白電腦操作系統中進程的知識,原理都是相同的。 系統根據PCB結構中的狀態值控制進程。 單CPU系統中,任一時刻處於執行狀態的進程只有一個。 進程的五種狀態:初始態,執行態,等待態,就緒態,終止態。 執行狀態:一個進程獲得了必要資源,並且在CPU上執行時 ...
  • 1. 檢測列表是否是空 沒必要去調用len方法去檢測一個列表是否是空,因為空列表求值就是False的。 可以用下麵的方法代替: 2. 在迭代列表的同時獲取元素的索引值 有時候你需要在迭代一個列表的同時獲取每個元素的索引值。通常的做法是: 更好的做法是: 3. 列表排序 創建一個persons列表: ...
  • DOS和Windows最大的不同在於DOS命令方式操作,所以使用者需要記住大量命令及其格式使用方法,DOS命令分為內部命令和外部命令, 內部 命令是隨每次啟動的COMMAND.COM裝入並常駐記憶體,而外部命令是一條單獨的可執行文件。在操作時要記住的是,內部命令在任何時候都可以使 用,而外部命令需要保 ...
  • 大型網站優化-memcache技術 memory+cache 記憶體緩存 memcache簡介 memcache是一套分散式的高速緩存系統,由LiveJournal的Brad Fitzpatrick開發,目前被許多網站使用以提升網站的訪問速度,尤其對於一些大型的、需要頻繁訪問資料庫的網站訪問速度提升效 ...
  • Queue簡介 queue是隊列容器,是一種“先進先出”的容器。 queue是簡單地裝飾deque容器而成為另外的一種容器。 #include <queue> 1.queue對象的預設構造 2.queue的push()與pop()方法 queue.push(elem); //往隊尾添加元素 queu ...
  • 在Java程式設計中經常會見到this的使用,this使得程式設計變得規範、簡單、靈活。但是在使用過程中,在不同場 合它的含義並不完全相同,使用不當還會出現錯誤, 本文對this的幾種用法和出現的問題進行了分析詳解。 關鍵詞:類;對象;this;成員變數;方法;構造方法 中,Java語言提供了豐富的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...