首頁 後端開發 C#.Net教程 C#實作一個最簡單的HTTP伺服器

C#實作一個最簡單的HTTP伺服器

Nov 24, 2016 am 11:48 AM
c# http 伺服器

 簡介

  本文用C#實現了一個最簡單的HTTP伺服器類,你可以將它嵌入到自己的專案中,或者也可以閱讀程式碼來學習關於HTTP協定的知識。

 背景

  高效能的WEB應用一般都架設在強大的WEB伺服器上,例如IIS, Apache, 和Tomcat。然而,HTML是非常靈活的UI標記語言,也就是說任何應用程式和後端服務都可以提供HTML的生成支援。在這個小小的例子中,像IIS,、Apache這樣的伺服器消耗的資源太大了,我們需要自己實作一個簡單的HTTP伺服器,將它嵌入到我們的應用中用來處理WEB請求。我們只需要一個類別就可以實現了,很簡單。

 程式碼實作

  首先我們來回顧一下如何使用類,然後我們再來分析實現的具體細節。這裡我們創建了一個繼承於HttpServer的類,並實現了handleGETRequest 和handlePOSTRequest  這兩個抽象方法:

public class MyHttpServer : HttpServer {
    public MyHttpServer(int port)
        : base(port) {
    }
    public override void handleGETRequest(HttpProcessor p) {
        Console.WriteLine("request: {0}", p.http_url);
        p.writeSuccess();
        p.outputStream.WriteLine("<html><body><h1>test server</h1>");
        p.outputStream.WriteLine("Current Time: " + DateTime.Now.ToString());
        p.outputStream.WriteLine("url : {0}", p.http_url);
 
        p.outputStream.WriteLine("<form method=post action=/form>");
        p.outputStream.WriteLine("<input type=text name=foo value=foovalue>");
        p.outputStream.WriteLine("<input type=submit name=bar value=barvalue>");
        p.outputStream.WriteLine("</form>");
    }
 
    public override void handlePOSTRequest(HttpProcessor p, StreamReader inputData) {
        Console.WriteLine("POST request: {0}", p.http_url);
        string data = inputData.ReadToEnd();
 
        p.outputStream.WriteLine("<html><body><h1>test server</h1>");
        p.outputStream.WriteLine("<a href=/test>return</a><p>");
        p.outputStream.WriteLine("postbody: <pre class="brush:php;toolbar:false">{0}
", data); } }
登入後複製

當開始處理一個簡單的請求時,我們需要單獨啟動一個線程來監聽一個端口,比如8080端口:

HttpServer httpServer = new MyHttpServer(8080);
Thread thread = new Thread(new ThreadStart(httpServer.listen));
thread.Start();
登入後複製

如果你編譯運行這個項目,你會在瀏覽器http://localhost:8080位址下看到頁面上產生的範例內容。讓我們來簡單來看看這個HTTP伺服器引擎是怎麼實現的。

  這個WEB伺服器由兩個元件構成,一個是負責啟動TcpListener來監聽指定連接埠的HttpServer類,並且用AcceptTcpClient()方法循環處理TCP連線請求,這是處理TCP連線的第一步。然後請求到達“已指定“的端口,接著就會創建一對新的端口,用來初始化客戶端到伺服器端的TCP連接。這對連接埠便是TcpClient的session,這樣就可以保持我們的主連接埠可以繼續接收新的連線請求。從下面的程式碼我們可以看到,每個監聽程式都會建立一個新的TcpClien,HttpServer類別又會建立一個新的HttpProcessor,然後啟動一個執行緒來操作。 HttpServer類別中還包含兩個抽象方法,你必須實作這兩個方法。

public abstract class HttpServer {
 
    protected int port;
    TcpListener listener;
    bool is_active = true;
 
    public HttpServer(int port) {
        this.port = port;
    }
 
    public void listen() {
        listener = new TcpListener(port);
        listener.Start();
        while (is_active) {               
            TcpClient s = listener.AcceptTcpClient();
            HttpProcessor processor = new HttpProcessor(s, this);
            Thread thread = new Thread(new ThreadStart(processor.process));
            thread.Start();
            Thread.Sleep(1);
        }
    }
 
    public abstract void handleGETRequest(HttpProcessor p);
    public abstract void handlePOSTRequest(HttpProcessor p, StreamReader inputData);
}
登入後複製

 這樣,一個新的tcp連接就在自己的線程中被HttpProcessor處理了,HttpProcessor的工作就是正確解析HTTP頭,並且控制正確實現的抽象方法。下面我們來看看HTTP頭的處理過程,HTTP請求的第一行程式碼如下:

GET /myurl HTTP/1.0
登入後複製

    在設定process()的輸入和輸出後,HttpProcessor就會呼叫parseRequest()方法。

public void parseRequest() {
    String request = inputStream.ReadLine();
    string[] tokens = request.Split(&#39; &#39;);
    if (tokens.Length != 3) {
        throw new Exception("invalid http request line");
    }
    http_method = tokens[0].ToUpper();
    http_url = tokens[1];
    http_protocol_versionstring = tokens[2];
 
    Console.WriteLine("starting: " + request);
}
登入後複製

 HTTP請求由3部分組成,所以我們只需要用string.Split()方法將它們分割成3部分即可,接下來就是接收和解析來自客戶端的HTTP頭訊息,頭資訊中的每一行資料是以Key-Value(鍵-值)形式保存,空行表示HTTP頭資訊結束標誌,我們程式碼中用readHeaders方法來讀取HTTP頭資訊:

public void readHeaders() {
    Console.WriteLine("readHeaders()");
    String line;
    while ((line = inputStream.ReadLine()) != null) {
        if (line.Equals("")) {
            Console.WriteLine("got headers");
            return;
        }
 
        int separator = line.IndexOf(&#39;:&#39;);
        if (separator == -1) {
            throw new Exception("invalid http header line: " + line);
        }
        String name = line.Substring(0, separator);
        int pos = separator + 1;
        while ((pos < line.Length) && (line[pos] == &#39; &#39;)) {
            pos++; // 过滤掉所有空格
        }
 
        string value = line.Substring(pos, line.Length - pos);
        Console.WriteLine("header: {0}:{1}",name,value);
        httpHeaders[name] = value;
    }
}
登入後複製

 到這裡,我們已經了解如何處理簡單的GET和POST請求,它們分別被分配給正確的handler處理程序。在本例中,發送資料的時候有一個棘手的問題需要處理,那就是請求頭資訊中包含發送資料的長度資訊content-length,當我們希望子類別HttpServer中的handlePOSTRequest方法能夠正確處理資料時,我們需要將資料長度content-length資訊一起放入資料流中,否則發送端會因為等待永遠不可能到達的資料和阻斷而等待。我們用了一種看起來不那麼優雅但非常有效的方法來處理這種情況,即將資料發送給POST處理方法前先把資料讀入到MemoryStream中。這種做法不太理想,原因如下:如果發送的資料很大,甚至是上傳一個文件,那麼我們將這些資料緩存在記憶體中就不那麼合適甚至是不可能的。理想的方法是限制post的長度,例如我們可以將資料長度限制為10MB。

  這個簡易版HTTP伺服器另一個簡化的地方就是content-type的回傳值,在HTTP協定中,伺服器總是會將資料的MIME-Type傳送給客戶端,告訴客戶端自己需要接收什麼類型的資料。在writeSuccess()方法中,我們看到,伺服器總是會傳送text/html類型,如果你需要加入其他的類型,你可以擴充這個方法。


本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora:巫婆樹的耳語 - 如何解鎖抓鉤
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1666
14
CakePHP 教程
1425
52
Laravel 教程
1325
25
PHP教程
1273
29
C# 教程
1252
24
使用 C# 的活動目錄 使用 C# 的活動目錄 Sep 03, 2024 pm 03:33 PM

使用 C# 的 Active Directory 指南。在這裡,我們討論 Active Directory 在 C# 中的介紹和工作原理以及語法和範例。

C# 中的隨機數產生器 C# 中的隨機數產生器 Sep 03, 2024 pm 03:34 PM

C# 隨機數產生器指南。在這裡,我們討論隨機數產生器的工作原理、偽隨機數和安全數的概念。

C# 資料網格視圖 C# 資料網格視圖 Sep 03, 2024 pm 03:32 PM

C# 資料網格視圖指南。在這裡,我們討論如何從 SQL 資料庫或 Excel 檔案載入和匯出資料網格視圖的範例。

C# 中的階乘 C# 中的階乘 Sep 03, 2024 pm 03:34 PM

C# 階乘指南。這裡我們討論 C# 中階乘的介紹以及不同的範例和程式碼實作。

c#多線程和異步的區別 c#多線程和異步的區別 Apr 03, 2025 pm 02:57 PM

多線程和異步的區別在於,多線程同時執行多個線程,而異步在不阻塞當前線程的情況下執行操作。多線程用於計算密集型任務,而異步用於用戶交互操作。多線程的優勢是提高計算性能,異步的優勢是不阻塞 UI 線程。選擇多線程還是異步取決於任務性質:計算密集型任務使用多線程,與外部資源交互且需要保持 UI 響應的任務使用異步。

C# 中的模式 C# 中的模式 Sep 03, 2024 pm 03:33 PM

C# 模式指南。在這裡,我們討論 C# 中模式的介紹和前 3 種類型,以及其範例和程式碼實作。

C# 中的質數 C# 中的質數 Sep 03, 2024 pm 03:35 PM

C# 質數指南。這裡我們討論c#中素數的介紹和範例以及程式碼實作。

xml怎麼改格式 xml怎麼改格式 Apr 03, 2025 am 08:42 AM

可以採用多種方法修改 XML 格式:使用文本編輯器(如 Notepad )進行手工編輯;使用在線或桌面 XML 格式化工具(如 XMLbeautifier)進行自動格式化;使用 XML 轉換工具(如 XSLT)定義轉換規則;或者使用編程語言(如 Python)進行解析和操作。修改時需謹慎,並備份原始文件。

See all articles