上篇说到当一个Http请求流到HttpHandler这里时才开始对它的处理,那么一个请求经过HttpHandler之后, 到底怎么对它处理呢,也就是说HttpHandler会触发哪些事件,触发的顺序如何,我们可以在此中间做些什么?

话说我们今天的重中之重:页面生命周期,说的就是Page类在处理页面的过程中都发生了哪些事件,而这些事件又是按照什么顺序发生的。ASP.NET的页面生命周期跟我们之前的理论讲解完全不同,它具有非常强大的实用性,所以我想通过一个小例子来进行解释和说明,以便让大家看起来更清晰和容易理解。因为Page类的ProcessRequest方法实现非常的复杂,我把代码反编译过来给大家看我觉得也没什么意义,所以我就着重给大家看一下整个页面的事件是以何种顺序被引发的,以及在这些事件中,我们能做什么。在这个过程中,我们主要关注如下问题:

1,    事件的执行顺序

2,    控件何时被初始化(我们什么时候能用它)

3,    ViewState何时可用

4,    更改MasterPage和Theme

5,    在各个事件中还能干什么工作

按照如下步骤建立示例:

一,我们创建一个website

二,加入两个MasterPage分别为site.master和map.master,分别在上面加一个label进行说明:this is site/map master page.

三,分别加入两个theme为BlueSkin和RedSkin,分别对button的背景色进行设置,一个是Blue,另外一个是Red

四,在default页面中加入两个label和一个button

五,Default页面代码如下:

private int step = ;
private string GetEventName(string eventName)
{
step += ;
return eventName;
} protected override void OnPreInit(EventArgs e)
{
base.OnPreInit(e);
Response.Write(GetEventName("pre init event, this is the " + step + "th step!"));
//lblMessage.Text = "on pre init"; this.MasterPageFile = "map.master";
this.Theme = "BlueSkin";
if (lblMessage2 == null)
{
Response.Write("<span style='color:red;'>Server control has not been initialed on pre init</span>");
}
else
{
Response.Write("<span style='color:red;'>Server control has been initialed here</span>");
}
if(this.ViewState.Count>)
{
Response.Write("ViewState can be used here, the ID is "+ViewState["ID"].ToString());
} Response.Write("<br/>");
}
protected override void OnInit(EventArgs e)
{
if (lblMessage2 == null)
{
Response.Write("<span style='color:red;'>Server control has not been initialed on pre init</span>");
}
else
{
Response.Write("<span style='color:red;'>Server control has been initialed here</span>");
}
base.OnInit(e); Response.Write(GetEventName("init event, this is the " + step + "the step! "));
if (lblMessage2 == null)
{
Response.Write("<span style='color:red;'>Server control has not been initialed on pre init</span>");
}
else
{
Response.Write("<span style='color:red;'>Server control has been initialed here</span>");
}
if (this.ViewState.Count > )
{
Response.Write("<span style='color:red;'>ViewState can be used here, the ID is " + ViewState["ID"].ToString() + "</span>");
}
Response.Write("<br/>"); //this.MasterPageFile = "map.master";
//this.Theme = "BlueSkin";
}
protected override void OnInitComplete(EventArgs e)
{
base.OnInitComplete(e);
Response.Write(GetEventName("init complete event, this is the " + step + "th step!"));
if (this.ViewState.Count > )
{
Response.Write("<span style='color:red;'>ViewState can be used here, the ID is " + ViewState["ID"].ToString() + "</span>");
}
else
{
Response.Write("<span style='color:red;'>ViewState can not be used here</span>");
}
Response.Write("<br/>");
}
protected override void OnPreLoad(EventArgs e)
{
if (this.ViewState.Count > )
{
Response.Write("<span style='color:red;'>ViewState can be used here, the ID is " + ViewState["ID"].ToString() + "</span>");
}
else
{
Response.Write("<span style='color:red;'>ViewState can not be used here</span>");
}
base.OnPreLoad(e);
Response.Write(GetEventName("pre load event, this is the " + step + "th step!"));
if (this.ViewState.Count > )
{
Response.Write("<span style='color:red;'>ViewState can be used here, the ID is " + ViewState["ID"].ToString() + "</span>");
}
else
{
Response.Write("<span style='color:red;'>ViewState can not be used here</span>");
}
Response.Write("<br/>");
}
protected void Page_Load(object sender, EventArgs e)
{
Response.Write(GetEventName("page load system provided, this is " + step + "th step!"));
if (this.ViewState.Count > )
{
Response.Write("<span style='color:red;'>ViewState can be used here, the ID is " + ViewState["ID"].ToString() + "</span>");
}
Response.Write("<br/>");
} protected override void OnLoadComplete(EventArgs e)
{
base.OnLoadComplete(e);
Response.Write(GetEventName("on load complete event, this is the " + step + "th step!<br/>"));
} protected override void OnPreRender(EventArgs e)
{
base.OnPreRender(e);
Response.Write(GetEventName("pre render event, this is the " + step + "th step!<br/>"));
} protected override void OnPreRenderComplete(EventArgs e)
{
base.OnPreRenderComplete(e);
Response.Write(GetEventName("pre render complete event, this is the " + step + "th step!<br/>"));
} protected override void OnSaveStateComplete(EventArgs e)
{ base.OnSaveStateComplete(e);
Response.Write(GetEventName("sae state complete event, this is the " + step + "th step!<br/>"));
} protected override void Render(HtmlTextWriter writer)
{
base.Render(writer);
Response.Write(GetEventName("render function, this is the " + step + "th step!<br/>"));
} protected override void OnUnload(EventArgs e)
{
if (this == null)
{
string aaa = string.Empty;
}
if (lblMessage2 == null)
{
string ga = string.Empty;
}
base.OnUnload(e);
if (this == null)
{ }
} protected override void OnLoad(EventArgs e)
{
base.OnLoad(e);
Response.Write(GetEventName("page load we created, this is the " + step + "th step!<br/>"));
}
protected void btnSubmit_Click1(object sender, EventArgs e)
{
ViewState["ID"] = "";
Response.Write(GetEventName("button control click event, this is the " + step + "th step!<br/>"));
}

执行结果:

七,代码分析

a)         从执行结果我们可以看到,事件的执行顺序为:

i.              PreInit

ii.              Init

iii.              InitComplete

iv.              PreLoad

v.              Load

vi.              Control Event(if they have)

vii.              LoadComplete

viii.              PreRender

ix.              PreRenderComplete

x.              SaveStateComplete

xi.              Render

xii.              Unload

b)         从结果中我们也可以看到,在PreInit事件发生后控件还没有被初始化,也就是我们还不能使用。而在Init刚刚发生的时候已经可以使用了。也就是说,控件初始化发生在PreInit之后Init之前。总而言之,我们关心的是我们最早可以在Init事件里对页面server端控件进行操控。

c)         关于ViewState,本示例也清晰的显示出它在Init Complete之后还是不能使用,而在PreLoad刚刚发生的时候,就已经可以使用了。

d)         通过对比Default.aspx页面的配置我们知道,在PreInit里面的Theme和MasterPage的设置是生效了的,我们可以在PreInit里设置Theme和MasterPage,而在其它任何位置设置这两个东西都将抛出异常(看我注视掉的代码)

e)         其它的事件:

Load: 这个事件可能是大家最熟悉的了。需要注意的是,Page对象会递归的调用子控件的onload事件直到页面和所有的子控件被加载完成。这个事件主要用来设置控件属性的值,建立数据库连接(通常不这么做)。

Control events: 这个就不多说了,主要是处理控件的事件,例如click。这也就让我们明白了每次我们click一个Button的时候,实际上是要先去执行load事件然后才执行click事件的,一般我们用!IsPostBack来判断一下从而避免执行不必要的加载逻辑。

LoadComplete: 页面所有的控件都被加载以后执行,暂时没有想到用来干什么。。。

PreRender: 在HTML被生成之前这是最后一个事件。每一个页面中的控件都有PreRender的过程。在这里对将要输出的HTML结果进行最后一次修改。

SaveStateComplete: 在这个时间发生之前,已经保存了所有控件和页面的,任何对page或者控件的改动都不会产生左右。暂时没想到用来干啥。

Render: 它不是一个事件而是一个方法。工作就是把HTML写回客户端浏览器。

UnLoad: 页面中的每一个控件都会发生这件事。在控件中,使用这个事件来做清理工作,例如关闭数据库连接等。对与页面本身也是做清理工作,例如关闭打开的文件和数据库连接,或者结束日志或者其它指定的工作。

f)          关于Unload事件,它实际上做的是销毁前的工作。在Unload事件发生以后,Page类被卸载和销毁,所以page类的字段值也就消失了,而我们通常也是使用在SaveStateComplete之前保存的ViewState来存储哪些我们想要存储的page里的数据。HttpRuntime做了销毁Page的动作,同样也是它创建的Page这个handler,现在我们知道原来不是HttpApplication雇佣了P_Handler, HttpApplication只是使用了它而已,真正雇佣(创建)并解雇(销毁)P_Handler的是老板HttpRuntime。

整个ASP.NET生命周期基本结束,如果大家想更深入的了解其中内情,请反编译System.Web.HttpRuntime类。如果您不喜欢研究那么深入,我想我这几篇文章应该可以对您有些帮助。

Asp.Net生命周期系列六的更多相关文章

  1. Asp.Net生命周期系列三

    上文讲到了HttpRunTime主要做了三个事情,我们先回忆一下. 第一:雇佣了项目经理(HttpApplication). 第二:建立了HttpModule列表,项目经理(HttpRunTime)就 ...

  2. Asp.Net生命周期系列四

    上回我们说的当一个Http请求来到HttpModule这里的时候,Asp.Net内部并未对这个Http请求做出任何的处理,我们可以对这个Http请求添加一些我们需要的信息,以方便我们控制这个Http请 ...

  3. Asp.Net生命周期系列五

    如果您看了我的前四篇文章,应该知道目前Http请求已经流到了HttpModule这个程序员手中了,而且我们可以注册自己的HttpModule并且可以在里面注册一些事件来控制这个Http请求,但是到目前 ...

  4. (转)Asp.Net生命周期系列五

    原文地址:http://www.cnblogs.com/skm-blog/p/3188697.html 如果您看了我的前四篇文章,应该知道目前Http请求已经流到了HttpModule这个程序员手中了 ...

  5. (转)Asp.Net生命周期系列三

    原文地址:http://www.cnblogs.com/skm-blog/p/3178862.html 上文讲到了HttpRunTime主要做了三个事情,我们先回忆一下. 第一:雇佣了项目经理(Htt ...

  6. Asp.Net生命周期系列二

    在上回书开始的时候我们提到博客园的IIS看了一眼我的请求后就直接交给ASP.NET去处理了,并且要求ASP.NET处理完之后返回HTML以供展示. 那么我们不仅要问: 1,    IIS肯定是没有眼睛 ...

  7. (转)Asp.Net生命周期系列一

    原文地址:http://www.cnblogs.com/skm-blog/archive/2013/07/07/3176713.html Asp.Net生命周期对于初级甚至中级程序员来说,一直都是一个 ...

  8. Asp.Net生命周期系列一

    Asp.Net生命周期对于初级甚至中级程序员来说,一直都是一个难题,很多程序员不了解生命周期,导致使用Asp.Net做开发感觉很不灵活,感觉太多东西被微软封装好了,我们不能改变,其实只要你稍微了解一下 ...

  9. Git使用总结 Asp.net生命周期与Http协议 托管代码与非托管代码的区别 通过IEnumerable接口遍历数据 依赖注入与控制反转 C#多线程——优先级 AutoFac容器初步 C#特性详解 C#特性详解 WPF 可触摸移动的ScrollViewer控件 .NET(C#)能开发出什么样的APP?盘点那些通过Smobiler开发的移动应用

    一,原理 首先,我们要明白Git是什么,它是一个管理工具或软件,用来管理什么的呢?当然是在软件开发过程中管理软件或者文件的不同版本的工具,一些作家也可以用这个管理自己创作的文本文件,由Linus开发的 ...

随机推荐

  1. 让UIScrollView、UITableView的滚动条一直显示

    先用xcode5.1.1或更低版本创建一个Category,如图: 然后拷贝以下代码到刚创建的UIImageView+ForScrollView.m文件中: - (void) setAlpha:(fl ...

  2. ASP.Net 验证控件 RegularExpressionValidator

    定义和用法 RegularExpressionValidator 控件用于验证输入值是否匹配指定的模式. 注释:除非浏览器不支持客户端验证或 EnableClientScript 属性被设置为 fal ...

  3. SharePoint 2010 配置基于MemberShip的身份验证

    场景:通常需要为sharepoint打通其他的系统整合到sharepoint认证,ad通常是为内部域用户,外网访问的可以使用membership来登录,那么这个既可以内部用户访问,外部用户也可以访问 ...

  4. 在Eclipse中导入文件和文件夹

    一. 将JAVA文件夹导入Eclipse中的方法: 方法一:直接将java文件夹复制,然后粘贴到项目下: 方法二:(1)打开Eclipse,点击项目的空白处,现在import: (2)现在Existi ...

  5. 【Mongodb】---关联表查询population

    Population MongoDB是非关联数据库.但是有时候我们还是想引用其它的文档.这就是population的用武之地. Population是从其它文档替换文档中的特定路径.我们可以迁移一个单 ...

  6. Java之简单图形面积计算

    设计一个类层次,定义一个抽象类--形状(类名:shape).其中包括求形状的面积抽象方法(方法名:calArea()).继承该抽象类定义三角形(Triangle).矩形(Rectangle).圆(Ci ...

  7. 第十篇、让UIScrollView的滚动条常显

    UIScrollView滚动条一直显示 1.我们知道滚动条是一个UIImageView, 滚动条隐藏是因为设置了alpha属性为0, 所有我们写一个UIImageView的分类 #define noD ...

  8. 20150313--AJAX

    from,提交信息,抛弃当前页面,并刷新页面. AJAX,从当前页面获取信息,并提交.反馈信息,不会抛弃页面. 例: 用户注册时,验证用户名是否已存在, 创建两个标准控件:TextBox1,Label ...

  9. 学习笔记:JavaScript传参方式———ECMAScript中所有函数的参数都是按值传递

    我们把命名参数(arguments)视为局部变量,在向参数传递基本类型值时,如同基本类型变量的复制一样,传递一个副本,参数在函数内部的改变不会影响外部的基本类型值.如: function add10( ...

  10. transport

    #include<iostream> using namespace std; int transport(int a) { ; ) ; else a=a/; d=; ) { a=a*+; ...