如何理解IIS 7的兩種應用程序池的管道模式(Managed Pipeline Mode)
之前我寫過一篇博客文章,講的是.NET 4.0的應用程序部署問題。有網友問到一個相關問題就是:如何理解IIS 7中的應用程序池的管道模式,尤其是如何理解托管模型(integrated mode),今天特意再寫一篇文章來介紹這個問題。
IIS 7是微軟最新版本的IIS版本,從Vista開始提供,目前在Vista,Windows 7,Windows Server 2008中提供。這個全新的版本中,一個突出的亮點就是,它提供了兩種管道模式,來支持不同的應用程序場景。
這里提到的管道模式,指的是應用程序池(Application Pool)的一個屬性
上圖中可以看到,這兩種管道模式分別為Integrated(集成)和 Classic(經典)
那么,到底如何理解這兩種模式呢?
Classic模式:指的是與IIS 6或者之前版本保持兼容的一種模式,一個典型問題就是,在處理ASP.NET這種動態網站的時候,它是通過一個所謂的ISAPI程序,作為插件的方式來工作的。針對不同的動態應用程序(例如ASP,PHP等),會需要不同的ISAPI。
例如,下面就是一個注冊號的ISAPI映射
從上圖可以看出,不同的Request, 指定了不同的ISAPI程序。下圖是對于這種Pipeline的一個圖形化說明
Integrated模式:這種全新的模式,允許我們將ASP.NET更好地與IIS集成,甚至允許我們在ASP.NET中編寫一些功能(例如Module)來改變IIS的行為(擴展)。集成的好處是,不再通過ISAPI的方式,提高了速度和穩定性。至于擴展,則可以使得我們對于IIS,以及其他類型的請求有更多的控制。(例如,我們希望靜態網頁也具備一些特殊的行為)
下圖解釋了這一點。
以上兩個圖片來自與下面這個文章,并且該文章有更詳細的一些理論介紹。
http://learn.iis.net/page.aspx/243/aspnet-integration-with-iis-7/
下面,我就通過一個例子,來幫助大家更好地了解集成模型。
這個例子里面,我有一個特殊的需求就是,我希望對網站里面所有請求做一個日志監控,不管是動態網頁,還是靜態網頁。
1. 創建一個Web Application
2. 添加一個HttpModule
為了對用戶請求進行監控,我們一般會編寫一個HttpModule。
我為該模塊實現簡單的功能(將用戶的請求地址打印出來在頁面上):
using System; using System.Web; namespace WebApplication2 { public class MyModule1 : IHttpModule { /// <summary> /// You will need to configure this module in the web.config file of your /// web and register it with IIS before being able to use it. For more information /// see the following link: http://go.microsoft.com/?linkid=8101007 /// </summary> #region IHttpModule Members public void Dispose() { //clean-up code here. } public void Init(HttpApplication context) { // Below is an example of how you can handle LogRequest event and provide // custom logging implementation for it context.LogRequest += new EventHandler(OnLogRequest); } #endregion public void OnLogRequest(Object source, EventArgs e) { //custom logging logic can go here var app = (HttpApplication)source; app.Response.Write(app.Request.Url.ToString()); } } }
3.注冊該模塊
模塊是需要注冊才能夠使用的。我們一般會想到以前的做法
<system.web> <compilation debug="true" targetFramework="4.0" /> <httpModules> <add name="test" type="WebApplication2.MyModule1,WebApplication2"/> </httpModules> </system.web>
但這樣注冊,會遇到一個錯誤
這個錯誤的意思是,LogRequest這個操作,是必須運行在集成模式下的。
那么,到底如何注冊成集成模式的模塊呢?
我們需要將配置文件修改成下面這樣
<?xml version="1.0"?> <!-- For more information on how to configure your ASP.NET application, please visit http://go.microsoft.com/fwlink/?LinkId=169433 --> <configuration> <system.web> <compilation debug="true" targetFramework="4.0" /> <!--<httpModules> <add name="test" type="WebApplication2.MyModule1,WebApplication2"/> </httpModules>--> </system.web> <system.webServer> <modules> <add name="test" type="WebApplication2.MyModule1,WebApplication2"/> </modules> </system.webServer> </configuration>
請注意,現在多了一個system.webServer的節,里面有一個modules的節,可以配置需要注冊的一些HttpModule。
因為是注冊為system.webServer的Module,所以,在visual studio中調試是沒有效果的。
我們需要將該應用程序發布到IIS,并且設置為integrated mode。
4. 發布到IIS
有很多辦法進行發布,我所推薦的是用deploy package的方式。
請注意,我們使用的Application Pool是integrated mode的。
發布之后,我們在瀏覽器中瀏覽首頁,我們會發現在底部會有一個特殊的輸出,就是我們當前請求的地址信息。這說明,我們那個Module已經在工作了。
5. 測試該模塊對于靜態頁面的支持
如果僅僅是上面這樣,我們似乎看不出這種模式到底有何優勢。我們以前不也是可以實現這樣的效果嗎?
請你主要的是,以前的HttpModule只能影響到動態網頁,例如我們的ASPX網頁,而對于靜態網頁(例如html),或者其他類型的動態網頁(例如php等)是無能為力的。
那么,現在這種模式下情況是怎樣的呢?
我們可以添加一個簡單的html頁面,放在網站根目錄下
然后,我們去請求該頁面
是不是很神奇呢?雖然是靜態網頁,但因為我們那個模塊是注冊在IIS里面的,它改變了IIS的行為,所以仍然會在頁面底部插入一段輸出。
6. 總結
希望上面這樣的例子可以幫助大家更好地理解Integrated mode。它是允許我們將代碼插入到IIS內核中,而不再通過ISAPI的方式。這將帶來更好的性能和擴展性。