天天看點

http請求處理過程,ihttpmodule,ihttphandler 處理流程

在以前的ASP時候,當請求一個*.asp頁面檔案的時候,這個HTTP請求首先會被一個名為inetinfo.exe程序所截獲,這個程序實際上就是www服務。截獲之後它會将這個請求轉交給asp.dll程序,這個程序就會解釋這個asp頁面,然後将解釋後的資料流傳回給用戶端浏覽器。其實ASP.DLL是一個依附在IIS的ISAPI檔案,它負責了對諸如ASP檔案,ASA等檔案的解釋執行,

-------------------------------------

ASP.NET的HTTP請求處理方法

當用戶端向web伺服器請求一個*.aspx的頁面檔案時,同asp類似,這個http請求也會被inetinfo.exe程序截獲(www服務),它判斷檔案字尾之後,把這個請求轉交給ASPNET_ISAPI.DLL而ASPNET_ISAPI.DLL則會通過一個Http PipeLine的管道,将這個http請求發送給ASPNET_WP.EXE程序,當這個HTTP請求進入ASPNET_WP.EXE程序之後,asp.net framework就會通過HttpRuntime來處理這個Http請求,處理完畢後将結果傳回給用戶端。

------------------------------------

當一個http請求被送入到HttpRuntime之後,這個Http請求會繼續被送入到一個被稱之為HttpApplication Factory的一個容器當中,而這個容器會給出一個HttpApplication執行個體來處理傳遞進來的http請求,而後這個Http請求會依次進入到如下幾個容器中:

HttpModule --> HttpHandler Factory --> HttpHandler

當系統内部的HttpHandler的ProcessRequest方法處理完畢之後,整個Http Request就被處理完成了,用戶端也就得到相應的東東了。

完整的http請求在asp.net framework中的處理流程:

HttpRequest-->inetinfo.exe->ASPNET_ISAPI.DLL-->Http Pipeline-->ASPNET_WP.EXE-->HttpRuntime-->HttpApplication Factory-->HttpApplication-->HttpModule-->HttpHandler Factory-->HttpHandler-->HttpHandler.ProcessRequest()

如果想在中途截獲一個httpRequest并做些自己的處理,就應該在HttpRuntime運作時内部來做到這一點,确切的說時在HttpModule這個容器中做到這個的。

----------------------------------------

-------------------------------------

系統本身的HttpModule實作一個IHttpModule的接口,當然我們自己的類也能夠實作IHttpModule接口,這就可以替代系統的HttpModule對象了。

ASP.NET系統中預設的HttpModule:

DefaultAuthenticationModule  確定上下文中存在 Authentication 對象。無法繼承此類。

FileAuthorizationModule  驗證遠端使用者是否具有通路所請求檔案的 NT 權限。無法繼承此類。

FormsAuthenticationModule 啟用 ASP.NET 應用程式以使用 Forms 身份驗證。無法繼承此類。

PassportAuthenticationModule  提供環繞 PassportAuthentication 服務的包裝。無法繼承此類。

SessionStateModule   為應用程式提供會話狀态服務。

UrlAuthorizationModule   提供基于 URL 的授權服務以允許或拒絕對指定資源的通路。無法繼承此類。

WindowsAuthenticationModule  啟用 ASP.NET 應用程式以使用 Windows/IIS 身份驗證。無法繼承此類

--------------------------------------

這些系統預設的HttpModule是在檔案machine.config中配置的,和我們開發時使用到的web.config的關系是:是在ASP.NET FRAMEWORK啟動處理一個Http Request的時候,它會依次加載machine.config和請求頁面所在目錄的web.config檔案,如果在machine中配置了一個自己的HttpModule,你仍然可以在所在頁面的web.config檔案中remove掉這個映射關系。

public class HelloWorldModule : IHttpModule
{
    public HelloWorldModule()
    {
    }    public String ModuleName
    {
        get { return "HelloWorldModule"; }
    }    // In the Init function, register for HttpApplication
    // events by adding your handlers.
    public void Init(HttpApplication application)
    {
        application.BeginRequest +=
            (new EventHandler(this.Application_BeginRequest));
        application.EndRequest +=
            (new EventHandler(this.Application_EndRequest));
    }    private void Application_BeginRequest(Object source,
         EventArgs e)
    {
    // Create HttpApplication and HttpContext objects to access
    // request and response properties.
        HttpApplication application = (HttpApplication)source;
        HttpContext context = application.Context;
        context.Response.Write("<h1><font color=red> HelloWorldModule: Beginning of Request</font></h1><hr>");
    }    private void Application_EndRequest(Object source, EventArgs e)
    {
        HttpApplication application = (HttpApplication)source;
        HttpContext context = application.Context;
        context.Response.Write("<hr><h1><font color=red>HelloWorldModule: End of Request</font></h1>");
    }    public void Dispose()
    {
    }
}    <system.web>
  <httpModules>
   <add name="HelloWorldModule" type="HelloWorldModule"/>
  </httpModules>
    </system.web>      

-----------------------------------------------------------------------------------

深入HttpModule

一個Http請求在被ASP.NET Framework捕獲之後會依次交給HttpModule以及HttpHandler來處理。hm與hh之間不是完全獨立的,實際上,http請求在hm傳遞的過程中會在某個事件内将控制權轉交給hh的,而真正的處理在HttpHandler中執行完成後,HttpHandler會再次将控制權交還給HttpModule

上面的代碼中的HttpModule的Init()中的參數是HttpApplication類型,它具有許多事件,包括BeginRequest,EndRequest,AuthentiacteRequest 等等。

-----------------------------------------------------------------

IHttpHandler

它是asp.net Framework提供的一個接口,定義了如果要實作一個Http請求的處理所需要必須實作的一些系統約定。也就是說,如果你想要自行處理某些類型的HTTP請求資訊流的話,你需要實作這些系統約定才能做到。譬如一個*.aspx檔案,用來處理此類型的Http請求,ASP.NET FRAMEWORK将會交給一個名為System.Web.UI.PageHandlerFactory的HttpHandler類來處理。

HH和HM一樣,系統會在最初始由ASP.NET FRAMEWORK首先加載machine.config中的HttpHandler,而後會加載Web應用程式所在目錄的web.config中的使用者自定義的HttpHandler類。但是系統與我們自定義的HH之間的關系是"覆寫"的,也就是說如果我們自定義了一個針對"*.aspx"的HttpHandler類的話,那麼系統會将對此http請求的處理權完全交給我們自己定義的這個HttpHandler類來處理,而我們自己的HttpHandler類則需要自己完全解析這個Http請求,并作出處理。

IHttpHandler接口中最重要的方法ProcessRequest,這個方法就是HttpHandler用來處理一個Http請求,當一個Http請求經過由HttpModule容器傳遞到HttpHandler容器中的時候,framework會調用HttpHandler的ProcessRequest方法來做對這個Http請求做真正的處理。

framework實際上并不是直接把相關頁面的HTTP請求定位到一個内部預設的IHttpHandler容器之上的,而是定位到了其 内部預設的IHttpHandler Factory上了。IHttpHandler Factory的作用就是對很多系統已經實作了的IHttpHandler容器進行排程和管理的,這樣做的優點是大大增強了系統的負荷性,提升了效率。

我們用Asp.Net做出來的Web通常是.Aspx字尾的.但是,你希望能有其它屬于自己的字尾嗎?我們一起來研究研究!      

      首先談談ASP.net的一些參數傳遞和頁面定向的方式

      第一,ASP.net是用Page.Navigate()調用新頁面的URL。Page.Navigate()向浏覽器傳回了一個http狀态碼302,使得浏覽器向server請求那個新的URL。這種導航方法導緻每次客戶請求都需兩次在client和server之間往返。

      第二,任何要傳遞到新頁面的資訊都需作為URL的參數或存儲在Session中或存儲在資料庫中以便新頁面得到這些資訊。傳統的ASP開發人員很習慣這種做法,但其他的web程式設計人員則有一些更進階的方法。但是很明顯的這兩個頁面是有依賴性的,而依賴性是編譯器捕捉不到的也是不容易在設計階段模組化的。是以在debug時,參數是否被正确的傳遞就隻有我們自己檢查了。再有傳統的資料傳遞方式有可能會暴露一些關鍵的資料。更為關鍵的是這使得面向對象的設計變得很複雜。

      但是我們可以自定義httphandler來擴充這種支援。

       要對HTTPMODULE和IHTTPHANDLER進行研究,必須先對ASP.NET的處理管道有一個了解。 

    在ASP.NET應用程式中,系統使用一組相關的類,通過一定的順序來處理用戶端的請求(REQUEST),ASP.NET應用程式的處理模式可稱之為HTTP處理管道。      

    HTTPMODULE和IHTTPHANDLER就是這個處理管道上的兩個處理環節。 HTTP處理管道中的類在SYSTEM.WEB名稱空間中定義,主要有以下類型: HTTPWORKERREQUEST 抽象類定義了ASP.NET頁面處理請求的基本方法; 

   HTTPRutime 提供了處理應用的一組服務; 

   HTTPContext 儲存了處理一次請求的所有相關上下文資訊; 

   HTTPApplicationFactory 提供相關目錄的應用程式; 

   HTTPApplication 定義了所有ASP.NET應用程式的通用的方法、屬性和事件。這個類也是在使用者在GLOBAL.ASAX檔案中定義的應用的基類; 

   Modules 處理請求前和響應後的事件; 

   Handlerfactories 提供應用程式中的HANDLERS; 

   Handlers 處理請求和響應。 

   在WINDOWS平台上,HTTP PIPLINE需要IIS的支援。為了運作ASP.NET應用,IIS需要以下兩個檔案:ASPNET_ISAPI.DLL和ASPNET_WP.EXE ASPNET_ISAPI.DLL是一個ISAPI EXTENTION他将發向IIS的請轉交ASPNET_WP.EXE處理 ASPNET_WP.EXE使用HTTPRUNTIME對請求進行具體處理 處理的過程可以用圖表示如下:

下面一個簡單的例子來了解!

SimpleHandler的工程,添加一個類MyHandler.cs,代碼如下:

namespace
  SimpleHandler
{
    
 public
  
 class
  MyHandler : IHttpHandler
    {
        
 public
  
 void
  ProcessRequest(HttpContext ctx)
        {
            HttpResponse response 
 =
  ctx.Response;
            
           response.Write(
 "
  My first Handler !
 "
 );
        }
        
 public
  
 bool
  IsReusable
        {
            
 get
  { 
 return
  
 true
 ; }
        }
    }
}      

繼續閱讀