陈斌彬的技术博客

Stay foolish,stay hungry

深入理解.net服务器控件

控件生命周期

1.初始化(对应Oninit方法)

这里页面通过ProcessRequest方法来递归遍历它的子控件(即页面中的空间,其实页面也算控件)。使子控件依次调用它们的Oninit方法。我们这里可以重写控件的OnInit方法,来扩展控件功能或增加初始化内容。在本阶段还要打开视图状态跟踪功能,调用TrackviewState方法,这样存储在viewState对象里面的值在页面回发时才能正确灰复到控件属性中。

2.加载视图状态(对应LoadViewState方法)

本阶段仅在页面回发时才执行,加载视图状态到控件。在第一次访问页面时我们还没有获得存储到视图状态的状态数据。

3.加载回传数据(对应LoadPostData方法)

在页面回发时执行。LoadPostData实现IPostBackDataHandler(实现控件数据回传必须要继承该接口)的一个方法,该方法参数NameValueCollection类型的对象装载了客户端提交的数据。另外该方法还会比较控件的旧值和新值返回一个bool类型值,以决定是否执行RaisePostDatachangedEvent方法。

客户修改窗体数据进行提交后,接收到的投递的数据是以 “&”符号隔开的一些键值串,页面处理器将投递的数据集合名称与页面控件的ID一一匹配,根据匹配ID检索对应的服务器控件有没有实现IPostBackDataHandler接口,如果实现了就调用控件的LoadPostData方法,给控件刷新其值。

4.装载(对应OnLoad方法)

页面装载时调用Page_Load()事件,在依次调用各个控件OnLoad方法.

5.数据回传事件通知(对应RaisePostDataChangeEvent方法)

在页面回发时执行。该方法也实现了IPostBackDataHandler接口。当bool值为true代表数据更改了,就执行该方法。

6.触发回发事件(对应RaisePostBackEvent方法)

在页面回发时执行。主要是处理引起回发的客户端事件,成功捕获回发的客户端事件进行服务器端的相应处理。也实现了IPostBackDataHandler接口。可以通过本法的参数来判断是哪个控件触发的回发事件,进而执行不同的事件处理逻辑。

7.预呈现(对应OnPreRender方法)

主要完成呈现(Render方法)之前所需要的一些处理。如注册JavaScript脚本和隐藏域控件等。

8.保存视图状态(对应SaveViewState方法)

该方法是把页面控件视图信息进行存储。第一次请求该页面就会执行该操作。

9.呈现(对应Render方法)

主要将控件标记和字符文本输出到服务器控件输出流中。

10.卸载(对应OnUnload方法)

对控件资源清除工作。

可以通过构造一个简单的服务器控件来熟悉它的生命周期

先创建一个控件库(新建项目选择ASP.NET服务器控件)和一个网站,再在类库中添加一个类(添加新建项选择ASP.NET 服务器控件)。在讲下列代码复制到类库中去。在生成该类库并在网站中引用该类库,再在该在网站上新建一个aspx文件,在页面工具箱中直接将控件拖到页面上即可。

 1 using System;
 2  using System.Collections.Generic;
 3  using System.ComponentModel;
 4 using System.Linq;
 5 using System.Text;
 6 using System.Web;
 7 using System.Web.UI;
 8 using System.Web.UI.WebControls;
 9 using System.Collections.Specialized;
10 
11 namespace ControlLibrary
12 {
13     [DefaultProperty("Text")]
14     [ToolboxData("<{0}:ControlLifecycle runat=server></{0}:ControlLifecycle>")]
15     public class ControlLifecycle : WebControl,IPostBackDataHandler,IPostBackEventHandler
16     {
17         protected override void OnInit(EventArgs e)
18         {
19             //输出
20             OutPut("1.OnInit");
21             base.OnInit(e);
22             //注册
23             this.Page.RegisterRequiresPostBack(this);
24         }
25 
26         protected override void LoadViewState(object savedState)
27         {
28             OutPut("2.LoadViewState");
29             base.LoadViewState(savedState);
30 
31         }
32 
33         public virtual bool LoadPostData(string postDataKey, NameValueCollection postCollection)
34         {
35 
36             OutPut("3.LoadPostData");
37             return true;
38         }
39 
40         protected override void OnLoad(EventArgs e)
41         {
42             OutPut("4.OnLoad");
43             base.OnLoad(e);
44         }
45 
46         public virtual void RaisePostDataChangedEvent()
47         {
48             OutPut("5. RaisePostDataChangedEvent");
49         }
50 
51         public virtual void RaisePostBackEvent(string eventArgument)
52         {
53             OutPut("6. RaisePostBackEvent");
54         }
55 
56         private void OutPut(string strText)
57         {
58             if (this.DesignMode == false)
59             {
60                 HttpContext.Current.Response.Write(strText + "<br>");
61             }
62         }
63 
64         protected override void OnPreRender(EventArgs e)
65         {
66             OutPut("7. OnPreRender");
67             base.OnPreRender(e);
68         }
69 
70         protected override object SaveViewState()
71         {
72             OutPut("8. SaveViewState");
73             base.SaveViewState();
74             return new Pair();
75         }
76 
77         protected override void Render(HtmlTextWriter writer)
78         {
79             writer.Write("<INPUT type=button name=\"{0}\" value=\"Click Me!\" style='position:absolute; left: 20; top: 280px' onclick=\"{1}\">", this.UniqueID, Page.ClientScript.GetPostBackEventReference(this, ""));
80             OutPut("9. Render");
81             base.Render(writer);
82         }
83    
84     }
85 }
86 

其实上面过程就算是我们开发了一个控件。了解它,最好的办法是尝试去开发它。

开发一个控件首先要选好它要继承的基类

选择基类

Control

控件开发基类,所有控件都直接或间接继承该类。提供了各类控件通用的属性和方法。

WebControl

webcontrol继承了control所有的属性,还增加了布局,可访问性,外观样式等特性。

CompositeControl

如果把现有控件聚合起来创建一个组合控件时,可以继承此类。该类默认实现了INamingContainer接口。

控件呈现顺序

上面介绍到了控件Render阶段主要是将控件标记和字符文本输出到服务器控件流当中。这里可以直接写HTML标记(上面例子就是的)也可以调用每个控件都有的RenderControl方法到输出流。在WebControl中,介绍以下5个函数

RenderControl(HtmlTextWriter writer)

Render(HtmlTextWriter writer)

RenderBeginTag(HtmlTextWriter writer)

RenderContents(HtmlTextWriter writer)

RenderEndTag(HtmlTextWirter writer)

在RenderControl 会内部调用Render,在Render会内部调用RenderBeginTag,RenderContents,RenderEndTag。

其中RenderBeginTag,RenderEndTag可以重写,默认值为<span></span>。它们是用来定义控件起始和结束的标记。

RenderContents看英文的意思就明白了。(待续。。。)

Resource Reference