学习吧少年...........................
修改中...........
当一次请求到达IIS
1、http.sys将请求发送到指定的应用程序池。
2、应用程序池再将请求交给池中的工作进程(w3wp.exe),
3、w3wp.exe根据请求URL的后缀,决定加载那个ISAPI扩展,ASP.NET映射(.aspx;.ashx;ascx;.asmx),如果是此后缀,那么要加载aspnet_isapi.dll
如果是静态的直接返回到HTTP.SYS 在浏览器中显示。
aspnet_isapi.dll扩展负责启动aspnet runtime,负责创建aspnet运行环境.还负责将请求交给ISAPIRuntime的PR方法,也就是非托管和托管程序的入口
.ISAPIRuntime对象 它调用了它的一个.ProcessRequest(ecb)方法; ecb是一个操作系统的句柄,指向了当前请求的内存空间,可以通过此句柄来拿到当前请求的报文;通过ecb句柄,创建了一个HttpWorkRequest对象.此对象就是对Http请求报文做了一些简单的封装.也就是请求的报文头,报文体而已;
4、一旦加载了aspnet_isapi.dll,则创建HttpRuntime类。
5、HttpRuntime类的ProcessRequest方法--再调用方法内的要触发的方法,最终创建HttpContex类。
6、之后HttpRuntime类请求HttpApplicationFactory类给返回一个HttpApplication对象。‘
7、HttpApplicationFactory类先检查是否先有空闲且可用的HttpApplication对象,如果有的的直接给。如果没有的话就要创建一个。
8、得到一个HttpApplication对象。。。。。
9、首先,HttpApplication操作---将Webconfig和Global.asax文件中的Httpmodule加载到HttpApplication的各种事件里。
10、执行HttpApplication的19个事件。。共有25个事件,而只有19个供程序员调用。
BeginRequest ---
AuthenticateRequest ---验证请求,开始检查用户的身份,一般获取请求的用户的信息。 (Authenticate意思为:验证)
PostAuthenticateRquest ---用户身份检查已经完成,检查完成后可以通过HttpContext的User属性获取到
AuthorizeRequest ---开始进行用户权限检查。如果用户没有通过上面的安全检查,一般直接会跳至EndRequest事件。
PostAuthorizeRequest ---用户请求已经获得授权。 (Authorize意思为:授权)
ResolveRequestCache ---如果存在以前处理的缓存结果,则不再进行请求的处理工作,直接返回缓存结果。
PostResolveRequestCache ---缓存检查结束
---根据请求资源的扩展名(在应用程序的配置文件中映射),选择并创建对应实现了IHttpHandler接口的处理类。
即:创建被请求页面的类对象,被请求页面继承子Page类,Page类实现了IHttpHandler接口。======被请求页面就是一个处理类
在第八个事件的时候,就创建了被请求前台页面类对象(前台页面类继承于其对应的后台页面类,后台页面类继承了Page类,Page类继承了TemplateControl类,TemplateControl类继承了Control类,Control类定义了许多重要的属性,例如ViewState属性。)。
PostMapRequestHandler ---已经创建处理请求的处理器对象(IHttpHandler)
AcquireRequestState ---获取请求状态,一般用于Session
PostAcquireRequestState ---已经获取了Session
请求管道的第十个事件中,会加载Session。首先会尝试将页面类对象转换成IPequiresSessionState接口对象,如果转换不成功就不加载Session对象,如果转换成功,则从请求报文中获得cookie里的SessionId,然后到服务器的Session池中,根据SessionId找出对应的Session对象,并将其引用赋值给页面对象的上下文对象的Session属性。
PreRuquestHandlerExecute ---准备执行处理程序。
即调用HttpHandler(即:被请求页面)的ProcessRequest方法。
----------------------------------------------即:被请求页面对象的ProcessRequest--------------------------------------------
先了解:被请求的页面对象是在第8个事件时被创建的。
被请求的页面就是一个HttpHandler(一个处理程序)
被请求的页面还是一个控件
过程介绍
在请求管道的第十一和十二个事件之间,被请求的页面就会调用了页面对象从父类继承来的ProcessRequest方法。在次方法中,首先调用了父类的FrameworkInitialize()方法,但因为被页面类重写了,所以执行的是当前页面类的FrameworkInitialize(),在中间就调用了_buildControlTree()方法来打造控件树。打造控件树就是将程序员在前台页面的所有代码转成控件Control类对象,然后添加到被请求的页面对象的Control集合中。
执行完_buildControlTree()方法后,就会调用ProcessRequestMain方法,在此方法里面就执行了整个页面生命周期(页面生命周期,其实就是调用了一系列的事件方法)。
过程执行 学习自:http://www.cnblogs.com/repository/archive/2010/08/13/1798334.html#2632930
private void ProcessRequest() { Thread currentThread = Thread.CurrentThread; CultureInfo currentCulture = currentThread.CurrentCulture; CultureInfo currentUICulture = currentThread.CurrentUICulture; try { this.ProcessRequest(true, true); //调用重载方法 } finally { this.RestoreCultures(currentThread, currentCulture, currentUICulture); } }
//我简化了代码 private void ProcessRequest(bool includeStagesBeforeAsyncPoint, bool includeStagesAfterAsyncPoint) { if (includeStagesBeforeAsyncPoint) { this.FrameworkInitialize(); //打造控件树。将程序员在前台页面的所有代码转成控件Control类对象,然后添加到被请求的页面对象的Control集合中。 } try { //这个才是重点,所有处理全在这个方法里 this.ProcessRequestMain(includeStagesBeforeAsyncPoint, includeStagesAfterAsyncPoint); } catch { throw; } }
View Code 简化了许多,去掉了一些与调试与跟踪的代码 private void ProcessRequestMain(bool includeStagesBeforeAsyncPoint, bool includeStagesAfterAsyncPoint) { try { HttpContext context = this.Context; string str = null; if (includeStagesBeforeAsyncPoint) { //预初始化,设置母版页与应用主题等操作
//准备开始初始化,此时模版中定义的控件已经创建,初始化页面的控件(以及页面本身) this.PerformPreInit(); //递归初始化,如果该控件包括子控件,则依次从内至外开始初始化(子控件->父控件) this.InitRecursive(null); //初始化完毕 this.OnInitComplete(EventArgs.Empty); //如果为回发 if (this.IsPostBack) { //对那些启用了视图状态的控件依次加载视图数据
//对页面返回的视图状态进行反序列化,还原视图状态 this.LoadAllState(); //处理回发数据。
//当页面提交Form时,框架将在每个提交数据的控件上实现IPostBackDataHandler接口。
//该方法在Page_Load前与Page_Load之后都要执行。 //在Page_Load之前执行时(即fBeforeLoad为true)时,它里面主要就是对那些实现了IPostBackDataHandler接口的控件
//处理Post数据,注册Post数据处理事件,其实也就是很常见的LoadPostData();
//通俗的说:将表单里提交的控件数据 设置 给控件树中的控件对应的属性=更新控件的状态
this.ProcessPostData(this._requestValueCollection, true);
} //终于快到页面加载了 this.OnPreLoad(EventArgs.Empty); //循环执行Load事件,方法循环执行控件的Load事件,如果控件包括子控件,则从外到内执行(父控件->子控件), //Page_Load事件在此执行 this.LoadRecursive(); if (this.IsPostBack) { //第二次处理回发数据,本次处理的控件上次没得到处理的剩下的控件 this.ProcessPostData(this._leftoverPostData, false); //这个方法主要是引发那些需要处理回发数据的控件在数据改变时需要引发的事件,如textbox控件的textchanged事件 //比如:postBackDataHandler.RaisePostDataChangedEvent(); this.RaiseChangedEvents(); //引起控件的回发事件,平时我们常用的控件的,如Button的click事件在这一步执行 this.RaisePostBackEvent(this._requestValueCollection); } //页面加载完毕 this.OnLoadComplete(EventArgs.Empty); } //验证 this._request.ValidateRawUrl(); //预呈现 this.PerformPreRenderComplete(); //保存视图状态 ,如: this.SavePageStateToPersistenceMedium(state); this.SaveAllState(); this.OnSaveStateComplete(EventArgs.Empty); //最后一步当就就是呈现了。 this.RenderControl(this.CreateHtmlTextWriter(this.Response.Output)); } catch (ThreadAbortException exception) { } catch (ConfigurationException) { } catch (Exception exception3) { } }
-----------------------------------------------------------------------------------------------------------------------------
PostRuquestHandlerExectue ---处理程序已经执行
ReleaseRequestState ---准备释放请求状态(Session)
PostReleaseRequestState ---已经释放了请求状态
UpdateRequestCache ---更新缓存
PostUpdateRequestCahe ---已经更新完缓存
EndRequest ---
PreSendRequestHeaders(.net 4.0 新增) ---可以根据发送的Header来动态设置一些参数,比如:通过 Content-Type 参数获知发送的
内容是 text/html网页,那么,可以通过启用输出的压缩来提高网络的传输速度。这个操作可以通过设置一个特殊的Header来通知浏览器。
PreSendRequestContent(.net 4.0 新增) ---如果配置了输出到客户端的压缩,那么可以在这个事件中包装输出到浏览器的流以实现输出的压缩。
此图取自:http://www.cnblogs.com/zhaoyang/archive/2011/11/16/2251200.html