分享
 
 
 

ASP.NET2.0服务器控件之自定义状态管理

王朝asp·作者佚名  2006-11-24
窄屏简体版  字體: |||超大  

在前面的系列文章中,我们曾经介绍了视图状态和控件状态的基本概念和典型应用,从中可以发现,视图状态和控件状态对于自定义服务器控件实现的重要性。本文将继续这一主题,重点介绍实现视图状态和控件状态自定义管理的方法。

自定义视图状态管理

在介绍视图状态时,我们曾经提到过:对于简单属性,例如,String、Int等类型,.NET执行引擎将自动启用默认视图状态管理机制,以便完成相应的功能。然而,如果开发人员在ViewState中保存的是自定义数据类型,或者需要实现自定义方式优化视图状态管理时,则必须实现自定义视图状态管理。

实现自定义视图状态管理可以通过两种方法。方法一:实现System.Web.UI命名空间中的IStateManager接口成员,其中包括IsTrackingViewState属性和TrackViewState、SaveViewState和LoadViewState方法。这种方法主要是针对自定义数据类型的视图状态管理的情况。方法二:重写Control基类的3个视图状态管理方法:TrackViewState、SaveViewState和LoadViewState。这些方法与IStateManager接口定义的3个方法名称一致。这种方法主要用于通过自定义方式优化默认视图状态管理的情况,其主要目的在于提高效率和性能。掌握以上两种实现方法的捷径是,必须深刻理解.NET框架内部实现视图状态管理的过程。下面两小节内容都是有关内部实现方法的介绍。每一节中均有实现代码,实际就相当于实例代码。所有服务器控件的自定义视图状态管理的实现都不会偏离那些代码所表达的逻辑。当读者真正掌握了那些内部实现方法,那么自定义视图状态管理的实现方法也就迎刃而解了。

1、实现基于IStateManager接口的自定义视图状态管理

对于复杂属性而言,多数需要实现自定义视图状态管理,其关键是实现System.Web.UI.IStateManager接口中定义的方法和属性。下面列举了IStateManager接口定义代码。

public interface IStateManager{ bool IsTrackingViewState {get;} void LoadViewState(object state); object SaveViewState(); void TrackViewState();}

如上代码所示,IStateManager接口要求类实现IsTrackingViewState属性,以及LoadViewState、SaveViewState和TrackViewState方法。IsTrackingViewState属性定义,当由类实现时,获取一个布尔值,通过该值指示服务器控件是否正在跟踪其视图状态更改。如果服务器控件正在跟踪其视图状态更改,则为true;否则为false。SaveViewState方法定义,当由类实现时,将服务器控件的视图状态更改保存到Object中。LoadViewState方法定义,当由类实现时,加载服务器控件以前保存的控件视图状态,其中的参数state表示包含控件保存的视图状态值的Object。TrackViewState方法定义,当由类实现时,指示服务器控件跟踪其视图状态更改。

ViewState属性与IStateManager接口之间存在密切联系。ViewState属性的类型是StateBag类,StateBag类通过实现IStateManager接口中定义的方法和属性来参与状态管理。其实现过程如下。

public sealed class StateBag : IStateManager, IDictionary,ICollection, IEnumerable{

private bool _isTrackingViewState;

private ArrayList _keys;

private ArrayList _values;

private StateItem _item;

bool IStateManager.IsTrackingViewState {

get { return _isTrackingViewState; }

}

void IStateManager.TrackViewState() {

_isTrackingViewState = true;

}

object IStateManager.SaveViewState() {

_keys = new ArrayList();

_values = new ArrayList();

IDictionaryEnumerator myDirctionaryEnumerator = this.GetEnumerator();

while(myDictionaryEnumerator.MoveNext()) {

if(this.Item[(String)myDictionaryEnumerator.Key].IsDirty) {

_keys.Add(myDictionaryEnumerator.Key);

_values.Add(myDictionaryEnumerator.Value);

}

}

if(_keys.Count>0) {

return new Pair(_keys,_values);

}

}

void IStateManager.LoadViewState(object savedState) {

if(savedState is Pair) {

_keys = (ArrayList)tempP.First;

_values = (ArrayList)tempP.Second;

IDictionaryEnumerator myDirctionaryEnumerator = this.GetEnumerator();

while(myDictionaryEnumerator.MoveNext()) {

for(int j=0;j<_keys.Count;j++)

{

if((String)myDictionaryEnumerator.Key == _keys[j].ToString());

{

this.Item[_keys[j].ToString()].Value = (object)_values[j];

}

}

}

}

}

}

请读者注意:以上代码为示意性代码,并非严格意义上的实现代码。在此列出,主要是用于说明StateBag类实现IStateManager接口的逻辑过程。

通过上面的代码,我们可以看到:

(1)在IsTrackingViewState属性中,将该属性设置为只读,并且使用私有变量_isTrackingViewState。

(2)在TrackViewState方法中,把IsTrackingViewState属性使用的私有变量_isTrackingViewState设置为true,这指示系统当某个StateItem添加到StateBag中,或者某个StateItem值被修改时,StateBag类就会自动将该StateItem标记为修改过即添加dirty标记。

(3)在SaveViewState方法中,循环StateBag中的每个StateItem,如果该StateItem被标记为dirty,那么就将其键和值分别添加到两个ArrayList中,并返回该对象。

(4)在LoadViewState方法中,执行了与SaveViewState方法相反的操作。首先将savedState对象分解为两个保存有键和值的ArrayList,然后将其中的值加载到相应的StateItem对象中。

以上就是ViewState属性实现IStateManager接口的基本过程。所有的视图状态管理过程,都要使用以上的实现过程,因此理解以上逻辑对于深入掌握自定义视图状态管理机制具有举足轻重的作用。

2、实现基于Control基类的自定义视图状态管理

如果开发人员需要优化默认视图状态管理机制,以提高控件运行效率和性能,那么必须理解Control基类中默认视图状态管理机制。通过掌握这个管理机制,可以模仿其处理过程以实现自定义视图状态管理。

实现基于Control基类的自定义视图状态管理,需要开发人员实现3个方法:LoadViewState、SaveViewState和TrackViewState。它们与上一小节中介绍的IStateManager接口成员方法同名,并且在方法意义上也基本相同。在此就不对这3个方法多做说明了。

Control基类中的默认视图状态管理机制定义了一个StateBag类型的ViewState属性,并将视图状态管理的任务委托给它。下面请看Control基类的默认状态管理的实现逻辑。

private StateBag _viewState;

protected virtual StateBag ViewState{

get {

if(_viewState != null)

{

return _viewState;

}

_viewState = new StateBag(ViewStateIgnoresCase);

if(IsTrackingViewState)

_viewState.TrackViewState();

return _viewState;

}

}

protected virtual void TrackViewState(){

if(_viewState != null) {

_viewState.TrackViewState();

}

return null;

}

protected virtual object SaveViewState(){

if(_viewState != null) {

_viewState.SaveViewState();

}

return null;

}

protected virtual void LoadViewState(object savedState){

if(savedState != null) {

ViewState.LoadViewState(savedState);

}

}

从上面的代码可以看出:ViewState属性是StateBag类型,当_viewState不为null时,则返回_viewState;当_viewState为null时,则初始化一个StateBag类型的变量_viewState,并判断控件是否正在跟踪其视图状态更改,如果服务器控件正在跟踪其视图状态更改,那么就调用TrackViewState方法开始状态跟踪,最后返回_viewState。另外,在TrackViewState、SaveViewState、LoadViewState方法中,均使用了StateBag类中有关视图状态管理的方法。

在Control基类的默认视图状态管理过程中,由于定义了ViewState属性为StateBag类型,所以必然使用上文中StateBag类实现视图状态管理的逻辑。如果让Control基类实现IStateManager接口中的方法和属性,那么其实现过程必然与StateBag类实现IStateManager接口大同小异,这必然将造成重复,由此可能造成.NET框架改变IStateManager接口的访问性质。此外,在进行自定义视图状态管理的过程中,可能出现StateBag类型与Control基类的视图状态管理机制的冲突,那样就可能产生混乱。而采用目前的这种方式,无论从灵活性、继承性、可复用性,以致从编程人员的习惯上,都具有很多优点。

自定义控件状态管理

视图状态与控件状态在数据管理方面不太相同。ASP.NET 2.0在支持针对简单属性的默认视图状态管理机制的同时,还支持自定义视图状态管理。然而,对于控件状态管理而言,则不存在默认控件状态管理机制。开发人员必须实现自定义控件状态管理过程。本小节介绍一下自定义控件状态管理的实现方法,这对于实现基于控件状态功能的对象有着重要意义。

实际上,在前面系列文章介绍控件状态的过程中,读者已经接触了控件状态实现的内容,其中就包括自定义控件状态管理的实现。这个过程的实现与基于Control基类的自定义视图状态管理非常相似,二者都需要重写Control基类中的方法。实现自定义控件状态管理需要重写Control基类的SaveControlState和LoadControlState。下面列举了一段简单示例代码。

public class Sample : Control {

private int currentIndex = 0;

protected override void OnInit(EventArgs e) {

Page.RegisterRequiresControlState(this);

base.OnInit(e);

}

protected override object SaveControlState() {

return currentIndex != 0 ? (object)currentIndex : null;

}

protected override void LoadControlState(object state) {

if (state != null) { currentIndex = (int)state; }

}

}

实现自定义控件状态管理分为3个关键步骤:

(1)调用RegisterRequiresControlState方法。该方法用于将自定义控件注册为具有持久性控件状态的控件。

(2)重写SaveControlState方法。该方法用于保存自页回发到服务器后发生的任何服务器控件状态更改。

(3)重写LoadControlState方法。该方法用于从SaveControlState方法保存的上一个页请求还原控件状态信息。

小结

本章主要介绍了自定义视图状态和控件状态管理的内容。它们对于实现自定义服务器控件有着重要意义。建议读者首先深入理解理论知识,然后再尝试进行开发工作,这样可获得更好的学习效果。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有