老生又长谈:HttpApplication,HttpModule,HttpContext及Asp.Net页生命周期

IIS在接到一个新的http请求后,最终会调用asp.net_isapi.dll的ISAPI扩展(特指IIS6.0环境,iis7.0的应用程序池默认为集成方式,相对有所变化),然后传递到httpRuntime Pipe(http运行时管道),Asp.Net这时才开始运行(即HttpRunTime是Asp.Net真正的入口),HttpRunTime会为每个asp.net应用自动创建一个HttpApplication的实例,而该实例中又包含以下属性:

注1

Application -->相当于传统意义上asp时代的application对象,通常用于定义一个asp.net应用的全局变量

Context -->HttpContext(上下文)类的实例【Asp.Net新增的】

Modules -->影响当前应用程序的HttpModule模块集合

Request -->类似于asp中的Request对象,通常用于接收一些特定的值(比如Request.Form或Request.QueryString)

Response -->类似于asp中的Response对象,通常用于向做页面输出指定内容(比如Resonse.Write)

Server -->类似于asp中的Server对象,通过它能获得一些服务端的信息(比如Server.MapPath)

Session -->类似于asp中的Session对象

User -->用于获取用户认证相关的安全信息

 

从上面的属性可以发现:很多其实在asp年代已在使用,只有Context,Modules,User这三个是Asp.Net新增的

 

HttpApplication类除了具备"注1"的几个属性外,还有自己的方法,这里特别提一下Init方法和Dispose方法,这二个方法均可重载.

 

它们的调用时机为:

Init方法在Application_Start之后调用,而Dispose在Application_End之前调用,另外Application_Start在整个asp.net应用的生命周期内只激发一次(比如IIS启动或网站启动时),类似的Application_End也只有当asp.net应用程序关闭时被调用(比如IIS停止或网站停止时)

 

除了Application_Start和Application_End方法,HttpApplication还提供了以下事件:

这些事件包括前面提到的可重载的Init及Dispose方法,再加上Session对应的Session_Start与Session_End方法,均可直接在Global.ascx.cs中以Application_XXX的形式使用(因为Global.ascx.cs中定义的类Global本身就是继承自HttpApplication的)

view source

public class Global : System.Web.HttpApplication

再来看一下相对asp而言,新增的Context,Modules,User这三个属性

 

Context

Context即HttpContext类的实例,在几乎整个aspx页面生命周期中,Context上下文一直伴随着各个环节向下传递

所以我们几乎可以在web应用中的任何环节,用HttpContext.Current来引用到当前的上下文实例,从HttpContext的定义上,还可以发现Context本身的属性中,又可以得到Application,ApplicationInstance,Profile,Response.Request...等对象的实例引用

回想一下:

view source

/// <summary>
///Handler1 的摘要说明
/// </summary>
public class Handler1 : IHttpHandler 
{public void ProcessRequest(HttpContext context){context.Response.ContentType = "text/plain";context.Response.Write("Hello World");}public bool IsReusable{get{return false;}}
}

我们在使用一个ashx文件时,ProcessRequest方法便是把当前上下文传递进来,进而通过context得到Response对象的引用,最终可以向页面输出任何想要的内容.

 

Modules:

每一个实现了IHttpModule接口的类,就可以被认为是Http模块组件,可以理解为http请求拦截器,拦截到http请求后,它能修改正在被处理的Context上下文,完事儿之后,再把控制权交还给管道,如果还有其它模块,则依次继续处理,直到所有Modules集合中的HttpModule完为止(注:可怜的http请求就这样给各个httpModuleX)

asp.net2.0默认内置了很多HttpModule,从Machine.Config文件中可以发现以下默认的内置模块:

2
AnonymouseIdentification --为匿名用户分配一个临时身份
FileAuthorization --
验证用户是否有请求文件的Windows NT许可
FormsAuthentication --
窗体身份验证模块(如果没有这个模块,asp.net就无法以用户名/密码[FOrms]方式验证)
OutputCache --
输出缓存模块
PassportAuthentication --PassPort
验证模块
Profile --
用户配置模块(如果没有它,asp.net中就无法使用Profile)
RoleManager --
角色管理
SessionSate --
会话状态模块
UrlAuthorization --
基于URL的身份验证模块
WindowsAuthentication --Windows
IIS身份验证模块

 

User

如果您使用过asp.net2.0内置的Membership/Role机制来进行访问认证,就会对User对象感到很熟悉,比如:

view source

if (HttpContext.Current.User.Identity.IsAuthenticated)
{//用户登录过了...
}

我们常用它来判断当前浏览用户的登录状态,关于User类的更详细定义,可参见MSDN


生命周期:

最后再来回顾一下Asp.Net中Page页的生命周期,Page中定义了几个事件:

总体上讲:一个ASPX页面被请求时,最终的生命周期就是由Page中定义的上述事件(还有一些可重载的回调方法)以及以前提到的HttpApplication类中定义的事件(以相应的回调方法)共同触发或调用,最终叠加形成的一连串处理过程。

如果先不考虑HttpApplication中的事件处理方法(即不考虑我们在Global.ascx.cs中定义的Application_XXX处理方法),Page中的事件(方法)常规触发(调用)顺序为:

01.Page_PreInit

02.Page_Init

03.Page_InitComplete

04.Page_PreLoad

05.Page_Load

06.Page_LoadComplete

07.Page_PreRender

08.Page_SaveStateComplete

09.Page_Unload

这是在Page页面未回发,且不考虑页面子控件的前提下正常的顺序,如果加入页面回发(比如在页面中放一个asp:Button,然后在Button的Click回发事件中加入处理函数)后,顺序稍微有些变化:

01.Page_PreInit

02.Page_Init

03.Page_InitComplete

04.Page_PreLoad

05.Page_Load

06.Button1_Click

07.Page_LoadComplete

08.Page_PreRender

09.Page_SaveStateComplete

10.Page_Unload

不同的地方在于:回发事件Button1_Click在Page_Load后被触发.

最后再把HttpApplication的事件考虑进来,看下叠加后的顺序,不过先别着急,我们先来看一种特殊情况,如果一个asp.net应用根目录下未设置默认页,这时直接浏览根目录,比如http://localhost:2345/ 时,Globl.ascx.cs中定义的Application_XXX方法的调用顺序如下:
2011-05-03 15:01:39 413 Application_Start

2011-05-03 15:01:39 491 Init

2011-05-03 15:01:39 491 Application_BeginRequest

2011-05-03 15:01:39 506 Application_AuthenticateRequest

2011-05-03 15:01:39 506 Application_PostAuthenticateRequest

2011-05-03 15:01:39 506 Application_AuthorizeRequest

2011-05-03 15:01:39 522 Application_PostAuthorizeRequest

2011-05-03 15:01:39 522 Application_ResolveRequestCache

2011-05-03 15:01:39 522 Application_PostResolveRequestCache

2011-05-03 15:01:39 522 Application_PostMapRequestHandler

2011-05-03 15:01:39 522 Application_AcquireRequestState

2011-05-03 15:01:39 537 Application_PostAcquireRequestState

2011-05-03 15:01:39 537 Application_PreRequestHandlerExecute

2011-05-03 15:01:39 553 Application_Error

2011-05-03 15:01:39 553 Application_EndRequest

2011-05-03 15:01:39 569 Application_PreSendRequestHeaders

2011-05-03 15:01:39 569 Application_PreSendRequestContent

可以看到会触发Application_Error事件,即HttpRuntime认为这是一个错误.

紧接着再浏览一个实际存在的页面,如果这时应用程序有严重错误,导致Application关闭(比如web.config配置错误),调用的顺序如下:
2011-05-03 15:03:47 704 Application_BeginRequest

2011-05-03 15:03:47 704 Application_AuthenticateRequest

2011-05-03 15:03:47 766 Application_PostAuthenticateRequest

2011-05-03 15:03:47 766 Application_AuthorizeRequest

2011-05-03 15:03:47 766 Application_PostAuthorizeRequest

2011-05-03 15:03:47 766 Application_ResolveRequestCache

2011-05-03 15:03:47 783 Application_PostResolveRequestCache

2011-05-03 15:03:48 667 Application_PostMapRequestHandler

2011-05-03 15:03:48 667 Application_AcquireRequestState

2011-05-03 15:03:48 683 Application_PostAcquireRequestState

2011-05-03 15:03:48 698 Application_PreRequestHandlerExecute

2011-05-03 15:03:48 745 Page_PreInit

2011-05-03 15:04:02 903 Page_Unload

2011-05-03 15:04:02 903 Application_Error

2011-05-03 15:04:02 918 Application_EndRequest

2011-05-03 15:04:02 996 Application_PreSendRequestHeaders

2011-05-03 15:04:02 996 Application_PreSendRequestContent

2011-05-03 15:04:03 371 Application_Disposed

2011-05-03 15:04:03 371 Dispose

2011-05-03 15:04:03 386 Application_End

对比刚才的顺序,会发现Application_Start及Init没有再次被调用,也印证了文章前面提到的一些结论(Application_Start在整个asp.net应用生命周期内只触发一次),而且从最后的三个输出能知道:应用程序关闭时Application_Disposed,Dispose,Application_End按顺序调用.

再"重新"浏览(指web Server重启)一下正常访问的页面,在不出错也不回发的情况下,顺序如下:
2011-05-03 15:08:11 513 Application_Start

2011-05-03 15:08:11 591 Init

2011-05-03 15:08:11 591 Application_BeginRequest

2011-05-03 15:08:11 591 Application_AuthenticateRequest

2011-05-03 15:08:11 591 Application_PostAuthenticateRequest

2011-05-03 15:08:11 606 Application_AuthorizeRequest

2011-05-03 15:08:11 606 Application_PostAuthorizeRequest

2011-05-03 15:08:11 606 Application_ResolveRequestCache

2011-05-03 15:08:11 606 Application_PostResolveRequestCache

2011-05-03 15:08:11 622 Application_PostMapRequestHandler

2011-05-03 15:08:11 637 Application_EndRequest

2011-05-03 15:08:11 637 Application_PreSendRequestHeaders

2011-05-03 15:08:11 637 Application_PreSendRequestContent

2011-05-03 15:08:11 637 Application_BeginRequest

2011-05-03 15:08:11 637 Application_AuthenticateRequest

2011-05-03 15:08:11 653 Application_PostAuthenticateRequest

2011-05-03 15:08:11 653 Application_AuthorizeRequest

2011-05-03 15:08:11 653 Application_PostAuthorizeRequest

2011-05-03 15:08:11 653 Application_ResolveRequestCache

2011-05-03 15:08:11 653 Application_PostResolveRequestCache

2011-05-03 15:08:11 653 Application_PostMapRequestHandler

2011-05-03 15:08:11 653 Session_Start

2011-05-03 15:08:11 653 Application_AcquireRequestState

2011-05-03 15:08:11 653 Application_PostAcquireRequestState

2011-05-03 15:08:11 653 Application_PreRequestHandlerExecute

2011-05-03 15:08:11 669 Page_PreInit

2011-05-03 15:08:11 684 Page_Init

2011-05-03 15:08:11 684 Page_InitComplete

2011-05-03 15:08:11 684 Page_PreLoad

2011-05-03 15:08:11 684 Page_Load

2011-05-03 15:08:11 684 Page_LoadComplete

2011-05-03 15:08:11 684 Page_PreRender

2011-05-03 15:08:11 684 Page_SaveStateComplete

2011-05-03 15:08:11 700 Page_Unload

2011-05-03 15:08:11 700 Application_PostRequestHandlerExecute

2011-05-03 15:08:11 700 Application_ReleaseRequestState

2011-05-03 15:08:11 700 Application_PostReleaseRequestState

2011-05-03 15:08:11 700 Application_UpdateRequestCache

2011-05-03 15:08:11 700 Application_PostUpdateRequestCache

2011-05-03 15:08:11 700 Application_EndRequest

2011-05-03 15:08:11 700 Application_PreSendRequestHeaders

2011-05-03 15:08:11 700 Application_PreSendRequestContent

2011-05-03 15:08:11 793 Application_BeginRequest

2011-05-03 15:08:11 793 Application_AuthenticateRequest

2011-05-03 15:08:11 793 Application_PostAuthenticateRequest

2011-05-03 15:08:11 793 Application_AuthorizeRequest

2011-05-03 15:08:11 793 Application_PostAuthorizeRequest

2011-05-03 15:08:11 793 Application_ResolveRequestCache

2011-05-03 15:08:11 793 Application_PostResolveRequestCache

2011-05-03 15:08:11 809 Application_PostMapRequestHandler

2011-05-03 15:08:11 809 Application_AcquireRequestState

2011-05-03 15:08:11 809 Application_PostAcquireRequestState

2011-05-03 15:08:11 809 Application_PreRequestHandlerExecute

2011-05-03 15:08:11 825 Application_PostRequestHandlerExecute

2011-05-03 15:08:11 825 Application_ReleaseRequestState

2011-05-03 15:08:11 840 Application_PostReleaseRequestState

2011-05-03 15:08:11 949 Application_UpdateRequestCache

2011-05-03 15:08:11 949 Application_PostUpdateRequestCache

2011-05-03 15:08:11 965 Application_EndRequest

2011-05-03 15:08:11 981 Application_PreSendRequestHeaders

2011-05-03 15:08:11 981 Application_PreSendRequestContent

哇!原来一个页面访问下来,会调用到这么多的方法,怪不得很多高并发的大型网站,通常都要自己写一个精减的HttpHandler用来取代Page做为基类,以期望获得更好的性能

更准确的页面生命周期解释,请查阅下面的文档,这是msdn官方网站对于Asp.Net页面生命周期的权威解释

文中源码 留下QQ号码,或加QQ:691647004 一起讨论

转载于:https://www.cnblogs.com/boyjsj/archive/2011/05/05/abouthttp.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/263515.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ByteBuffer常用方法详解

缓冲区(Buffer)就是在内存中预留指定大小的存储空间用来对输入/输出(I/O)的数据作临时存储&#xff0c;这部分预留的内存空间就叫做缓冲区&#xff1a; 使用缓冲区有这么两个好处&#xff1a; 1、减少实际的物理读写次数 2、缓冲区在创建时就被分配内存&#xff0c;这块内存区域…

LeetCode—221. 最大正方形

221. 最大正方形 题目描述&#xff1a; 在一个由 ‘0’ 和 ‘1’ 组成的二维矩阵内&#xff0c;找到只包含 ‘1’ 的最大正方形&#xff0c;并返回其面积。 考察重点&#xff1a;动态规划 方法概括&#xff1a;二维矩阵中查询最大矩形 dp[i][j] Min(dp[i-1][j],dp[i][j-1]…

Egret入门学习日记 --- 第二篇

第二篇&#xff08;学习篇&#xff09; 既然选好了Egret&#xff0c;那我就要想想怎么学了。 开始第一步&#xff0c;先加个Q群先&#xff0c;这不&#xff0c;拿到了一本《Egret HTML5游戏开发指南》&#xff0c;阅至三章&#xff0c;得到印象相对较深的好处和坏处&#xff1…

辨别文件的真实类型

2019独角兽企业重金招聘Python工程师标准>>> 参考博客http://blog.csdn.net/shixing_11/article/details/5708145 每个文件类型都对应着一个编码头部 下面这些是已知的文件头部[自定义的枚举类型] package org.masque.file; /*** * Description:文件对应的文件头,资…

ng build --prod --aot打包Angluar4项目报javaScript heap out of memory,内存溢出

这里 Allocation failed - JavaScript heap out of memory JavaScript堆内存不足&#xff0c;我们都知道 Node 是基于V8引擎&#xff0c;在一般的后端开发语言中&#xff0c;在基本的内存使用上没有什么限制&#xff0c;但是我去查阅了相关的资料才发现&#xff0c;在 Node 中通…

GARFIELD@12-02-2004

your BELLY is more ready to be convinced than your BRAIN 转载于:https://www.cnblogs.com/rexhost/archive/2004/12/02/72000.html

LeetCode—223. 矩形面积

223. 矩形面积 题目描述&#xff1a; 给你 二维 平面上两个 由直线构成且边与坐标轴平行/垂直 的矩形&#xff0c;请你计算并返回两个矩形覆盖的总面积。 每个矩形由其 左下 顶点和 右上 顶点坐标表示&#xff1a; 第一个矩形由其左下顶点 (ax1, ay1) 和右上顶点 (ax2, ay2)…

GARFIELD@01-31-2005

savage sandwich 转载于:https://www.cnblogs.com/rexhost/archive/2005/01/31/99836.html

【ABAP系列】SAP ABAP模块-任意report作为附件以邮件形式发送

公众号&#xff1a;SAP Technical本文作者&#xff1a;matinal原文出处&#xff1a;http://www.cnblogs.com/SAPmatinal/ 原文链接&#xff1a;【ABAP系列】SAP ABAP模块-任意report作为附件以邮件形式发送前言部分 大家可以关注我的公众号&#xff0c;公众号里的排版更好&…

[Mac]一些命令技巧

Git相关 mac下git默认不区分大小写&#xff0c;通过下面脚本可以改变 #!/bin/bash# 让git区分大小写 cd path-of-project git config core.ignorecase false git不会将空文件夹添加到版本控制中&#xff0c;下面脚本可以让空文件夹加到git # 将所有空文件夹添加到git cd path-o…

GARFIELD@02-24-2005

a cat on a ball 转载于:https://www.cnblogs.com/rexhost/archive/2005/02/25/109153.html

网站总结 和 相关控件 总结

2019独角兽企业重金招聘Python工程师标准>>> 学习篇 https://github.com/Aufree/trip-to-iOS 开源库 http://github.ibireme.com/github/list/ios/ 项目管理和工具 测试&#xff1a;TestFlight 依赖管理&#xff1a;CocoaPods 自动生成推送证书&#xff1a;Gen…

西门子新款A系列手机【ZZ】

linked from http://www.tompda.com/neirong.asp?id809 西门子新款A系列手机   首页 > 新闻资讯 提交 shizhi 2005-2-25  阅读:2739次 10篇评论 西门子公司近日推出了A系列手机&#xff0c;A系列手机作为入门级手机&#xff0c;设计和功能十分简洁和实用。 西门子A70:设…

Chord算法

转自&#xff1a;http://blog.csdn.net/wangxiaoqin00007/article/details/7374833 虽然网上搜索CHord&#xff0c;一搜一大堆&#xff0c;但大多讲得不太清楚明白。今天发现一篇blog&#xff0c;图文并茂&#xff0c;逻辑清楚且易懂&#xff0c;特意转载收藏。 P2P的一个常见问…

地籍宗地出图(一)

在ArcGIS中&#xff0c;我们经常需要标注&#xff1b;在地籍处理中&#xff0c;需要承包地出图。具体的出没有可参考的标准。通常软件中的做法有&#xff1a; 第一种&#xff0c;以村组为背景&#xff0c;将某户的承包地选择出来&#xff0c;并标黑。这样的结果&#xff0c;领导…

黑马程序员_java基础笔记(15)...银行业务调度系统_编码思路及代码

—————————— ASP.NetAndroidIOS开发、.Net培训、期待与您交流&#xff01;—————————— 1&#xff0c;面试题目&#xff1a;银行业务调度系统 模拟实现银行业务调度系统逻辑&#xff0c;具体需求如下&#xff1a; 银行内有6个业务窗口&#xff0c;1 — 4号窗口…

一次非常有意思的sql优化经历

原文:一次非常有意思的sql优化经历场景 我用的数据库是mysql5.6&#xff0c;下面简单的介绍下场景 课程表 create table Course(c_id int PRIMARY KEY,name varchar(10))数据100条 学生表: create table Student(id int PRIMARY KEY,name varchar(10))数据70000条 学生成绩表SC…

python matplotlib绘图大全(散点图、柱状图、饼图、极坐标图、热量图、三维图以及热图)...

//2019.7.14晚matplotlib七种常见图像输出编程大全 七种图形汇总输出如下&#xff1a; import numpy as np #导入数据结构nmupy模块import matplotlib.pyplot as plt #导入matplotlib图像输出模块plt.rcParams["font.sans-serif"]["SimHei"] #输出图像的标…

【活动】畅想云端加油站,赢iPad

2019独角兽企业重金招聘Python工程师标准>>> 中石化联手阿里云升级石油化工业务&#xff0c;已运行2月 中石化的“互联网”战略正在不断深化。4月20日消息&#xff0c;中石化与阿里云共同宣布&#xff0c;双方将展开技术合作&#xff0c;借助阿里巴巴在云计算、大数…

编写 Servlet 2.3 Filter

Servlets Filter 是Servlet 2.3 规范中新增加的&#xff0c;它是截取用户从客户端提交的请求&#xff0c;在还没有到达需要访问的资源时运行的一个类。它操纵来自客户端的请求&#xff0c;在资源还没有初发送到客户端前截取响应&#xff0c;并处理这些还没有发送到客户端的响应…