ASP.NET 2.0執行原理及其過程簡要分析
System.Web.HttpRuntime
System.Web.HttpApplicationFactory
System.Web.HttpApplication
System.Web.Compilation.BuildManager
System.Web.Compilation.ApplicationBuildProvider
System.Web.Compilation.BuildProvidersCompiler
System.Web.UI.PageHandlerFactory
閱讀建議:
用Reflector工具邊檢視ASP.NET 2.0的原始碼邊閱讀。
分析:
當我們通過瀏覽器向ASP.NET 2.0網站的一個asp.net頁面發起請求時,在伺服器端首先是IIS收到請求,IIS一看是asp.net頁面,心裡很開心,因為這個請求不用它處理,交給ASP.NET ISAPI就行了。ASP.NET ISAPI的工作也比較輕鬆,他的主要任務就是安排aspnet_wp.exe處理請求,並監視aspnet_wp.exe程式的執行情況,如果aspnet_wp.exe程式太累了,不能出色地完成任務,ASP.NET ISAPI就要讓他下崗,換一個新的aspnet_wp.exe來處理工作。
aspnet_wp.exe的主要任務是將請求交給一系列稱為的 HTTP 管道的託管物件。如果把ASP.NET ISAPI比做銷售經理,那aspnet_wp.exe就是生產經理,而HTTP 管道就是生產的流水線。負責流水線的小組就是HttpRuntime,生產經理aspnet_wp.exe會將訂單(HTTP請求)交給HttpRuntime小組的工作人員ProcessRequest(HttpWorkerRequest wr),HttpRuntime根據內部的分工,最終由ProcessRequestInternal(HttpWorkerRequest wr)在流水線上進行生產,所以ProcessRequestInternal(HttpWorkerRequest wr)是我們分析的重點。
ProcessRequestInternal的主要工作是:
1. 建立HttpContext例項。
2. 對第一次請求進行初始化(EnsureFirstRequestInit)。
a) 在EnsureFirstRequestInit中通過呼叫System.Web.HttpRuntime.FirstRequestInit進行一些初始化工作,比如:將Web.Config配置讀到到RuntimeConfig中,從bin目錄中裝載所有dll檔案。
3. 建立HttpWriter例項。
4. 通過呼叫HttpApplicationFactory.GetApplicationInstance建立HttpApplication例項。
在HttpApplicationFactory.GetApplicationInstance中有三個關鍵方法:
HttpApplicationFactory._theApplicationFactory.EnsureInited();
HttpApplicationFactory._theApplicationFactory.EnsureAppStartCalled(context);
HttpApplicationFactory._theApplicationFactory.GetNormalApplicationInstance(context);
下面我們對這三個方法逐個進行分析:
1) HttpApplicationFactory._theApplicationFactory.EnsureInited();
該方法檢查HttpApplicationFactory是否被初始化,如果沒有,就通過HttpApplicationFactory.Init()進行初始化。
在Init()中,先獲取global.asax檔案的完整路徑,然後呼叫CompileApplication()對global.asax進行編譯。
那編譯是如何進行的呢?
編譯的工作由BuildManager完成的。BuildManager先得到GlobalAsaxType(也就是HttpApplication),然後呼叫BuildManager.GetGlobalAsaxBuildResult()=》GetGlobalAsaxBuildResultInternal()=》EnsureTopLevelFilesCompiled()進行編譯。
在EnsureTopLevelFilesCompiled中,先進行CompilationStage.TopLevelFiles編譯,對下面三個目錄中的檔案進行編譯:
a. CompileResourcesDirectory();
編譯App_GlobalResources目錄。
b. CompileWebRefDirectory();
編譯App_WebReferences目錄。
c. CompileCodeDirectories();
編譯App_Code目錄。
接著進行CompilationStage.GlobalAsax 編譯,對global.asax進行編譯,方法呼叫情況:CompileGlobalAsax()=》ApplicationBuildProvider.GetGlobalAsaxBuildResult(BuildManager.IsPrecompiledApp)。
在GetGlobalAsaxBuildResult中具體的編譯是由ApplicationBuildProvider與BuildProvidersCompiler共同完成的。
BuildProvidersCompiler.PerformBuild();進行編譯工作。
ApplicationBuildProvider.GetBuildResult得到編譯的結果。
編譯成功後,會在C:\WINDOWS\Microsoft.NET\Framework\v2.0.50727\Temporary ASP.NET Files\相應的目錄中生成類似App_global.asax.mlgx7n2v.dll的dll檔案。
編譯生成的類名為ASP.global_asax,繼承自HttpApplication。
注:如果Web目錄中沒有Global.asax檔案,就不會編譯生成App_global.asax.mlgx7n2v.dll這樣的檔案。
2) HttpApplicationFactory._theApplicationFactory.EnsureAppStartCalled(context);
建立特定的HttpApplication例項,觸發ApplicationOnStart事件,執行ASP.global_asax中的Application_Start(object sender, EventArgs e)方法。這裡建立的HttpApplication例項在處理完事件後,就被回收。
3) HttpApplicationFactory._theApplicationFactory.GetNormalApplicationInstance(context);
該方法建立HttpApplication例項並進行初始化(呼叫System.Web.HttpApplication. InitInternal()方法)。
建立HttpApplication例項是根據實際的_theApplicationType進行建立。如果Web目錄中沒有global.asa檔案,也就是說沒有動態編譯生成ASP.global_asax型別,那就直接例項化HttpApplication。如果建立了ASP.global_asax型別,那就對ASP.global_asa進行例項化。
建立HttpApplication例項之後就是呼叫例項的InitInternal方法。
InitInternal方法也是我們重點分析的方法,該方法的主要功能如下:
1. InitModules():根據Web.Config的設定,建立相應的HttpModules。
2. HookupEventHandlersForAppplicationAndModules:根據發生的事件,呼叫HttpApplication例項中相應的事件處理函式。
3. 建立很多實現IExecutionStep介面的類的例項並新增到當前HttpApplication例項的_execSteps中,等待回撥時執行。從這裡我們可以看到HttpApplication是以非同步的方式處理請求,對請求的很多處理工作都放入了_execStep等待回撥時執行。
_execStep中主要的處理工作如下:
1) 對請求的路徑進行安全檢查,禁止非法路徑訪問(ValidatePathExecutionStep)。
2) 如果設定了UrlMappings, 進行RewritePath(UrlMappingsExecutionStep)。
3) 執行事件處理函式,比如:BeginRequest、AuthenticateRequest等等。
4) 獲取處理當前請求的HttpHandler,ASP.NET頁面的執行時編譯也是在這裡進行的。(MapHandlerExecutionStep)
該處理是通過呼叫System.Web.HttpApplication. MapHttpHandler方法。
在MapHttpHandler中,首先根據訪問的地址從web.config獲取相應的實現IHttpHandlerFactory的型別。對於asp.net頁面,預設是PageHanlderFactory。然後建立PageHanlderFactory例項,呼叫GetHandlerHelper,在GetHandlerHelper中呼叫BuildManager.CreateInstanceFromVirtualPath編譯並建立當前請求的ASP.NET頁面的例項(如果已經編譯過,直接從快取中載入)。
CreateInstanceFromVirtualPath經過幾次方法呼叫,將編譯任務給了BuildManager. CompileWebFile()。CompileWebFile從web.config得到相應的BuildProvider,對於.aspx檔案,相應的BuildProvider是PageBuildProvider。PageBuildProvider是如何進行頁面編譯的,這裡就不再就進一步分析了,如果你感興趣,可以進一步研究ASP.NET 2.0的原始碼。
5) 呼叫相應HttpHandler的.ProcessRequest方法處理請求(如果是非同步方式,呼叫BeginProcessReques)。(CallHandlerExecutionStep)
6) 將響應內容寫入Filter。(CallFilterExecutionStep)
5. 呼叫HttpApplication例項的BeginProcessRequest非同步處理請求。
上面所講的_execSteps中所發生的許多事情,都是在HttpRuntime呼叫HttpApplication BeginProcessRequest之後,在BeginProcessRequest中呼叫ResumeSteps後執行的。
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/12639172/viewspace-580657/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- crtmpserver 執行過程簡明分析Server
- ASP.NET執行過程- -ASP.NET
- sql執行過程分析SQL
- HiveSQL的執行過程分析HiveSQL
- 磁碟原理簡要分析
- 程式碼精簡執行過程
- Informix 執行緒sleep 分析過程ORM執行緒
- ASP.NET 執行原理ASP.NET
- hashCode()方法原始碼執行簡要分析原始碼
- 原始碼分析OKHttp的執行過程原始碼HTTP
- knockout原始碼分析之執行過程原始碼
- Mapreduce(二):MR的執行過程分析
- ORACLE關於執行計劃的簡要分析Oracle
- Java 執行緒池的建立過程分析Java執行緒
- mybatis-spring sql執行過程分析MyBatisSpringSQL
- 【基礎篇基本原理】sql執行過程SQL
- browserify執行原理分析
- Locust 執行原理分析
- sql 執行過程SQL
- 【DDL】DDL執行過程及其隱式COMMIT行為的探查與演示MIT
- laravel 應用層執行過程原始碼分析Laravel原始碼
- PostgreSQL的insert語句執行過程分析SQL
- 程式執行過程記憶體分析詳解記憶體
- 線上定時指令碼執行慢,分析過程指令碼
- JVM 執行緒堆疊分析過程詳解JVM執行緒
- Java 執行緒池執行原理分析Java執行緒
- JVM執行原理及Stack和Heap的實現過程JVM
- 【協程原理】 - 協程不過是使用者態的執行緒執行緒
- 從簡單程式碼入手,分析執行緒池原理執行緒
- ASP.NET MVC 幾種 Filter 的執行過程原始碼解析ASP.NETMVCFilter原始碼
- 原始碼簡析Spring-Integration執行過程原始碼Spring
- 一個簡單java程式的執行全過程Java
- SAP Spartacus 裡的 cxOutletRef 指令執行過程分析
- IP欺騙原理與過程分析
- Oracle DDL 執行過程Oracle
- Java 程式執行過程Java
- 指令的執行過程
- 簡單分析ThreadPoolExecutor回收工作執行緒的原理thread執行緒