是否是ASP.NET的CheckBoxList的Bug?
2010-04-29 02:22 横刀天笑 阅读(2442) 评论(11) 编辑 收藏 举报缘起
今天其他项目的同事碰到一个bug,封装的一个控件有些问题。先就描述一下这个控件。
控件是从ASP.NET自身的CheckBoxList派生而来的,然后扩展一些功能,控件最后样式如下图所示:
点击展开按钮后,在控件下方显示一个浮动层,里面放着一个CheckBoxList:
(暂时没有控件的真实截图,暂且对付着看吧,中间有黑点的表示选中)
给该控件扩展了一个事件,当点击展开的时候触发该事件,回发到服务器端,从数据库里读取数据,然后决定哪些值选中。代码示例:
1: public class DropDownListEx : CheckBoxList
2: {
3: public event EventHandler<EventArgs> Expanded;
4:
5: protected virtual OnExpanded(EventArgs e)
6: {
7: if(Expanded != null)
8: Expanded(this,e);
9: }
10:
11: protected override void OnPreRender(EventArgs e)
12: {
13: base.OnPreRender(e);
14: //注册回发,为扩展事件
15: if (this.Page != null && this.Enabled)
16: this.Page.RegisterRequiresPostBack(this);
17: }
18:
19: protected override bool LoadPostData(string postDataKey, NameValueCollection postCollection)
20: {
21: //判断按钮是否点击了,模拟的
22: if (postCollection["data"] == "click")
23: {
24: //触发Expanded事件
25: OnExpanded(new EventArgs());
26: return true;
27: }
28: else
29: return base.LoadPostData(postDataKey,postCollection);
30: }
31: }
下面是Expanded事件处理器代码示例:
1: this.DropDownListEx1.Expanded += new EventHandler<EventArgs>(DropDownListEx1_Expanded);
2:
3: private void DropDownListEx1_Expanded(object sender,EventArgs e)
4: {
5: //全部选中
6: foreach(ListItem item in this.DropDownListEx1.Items)
7: item.Selected = true;
8: }
表面看这段代码好像没什么问题,貌似也“一直”工作的很好,但是有用户突然发现,最后一个复选框“值3”,如果原来没有选中,即使在Expanded事件里,将所有复选框都选中,但回发完成后这最后一个依然是未选中状态。
最后调试发现,一个现象,DropDownListEx的LoadPostData多次调用,调用的顺序(语言不准确)是:值1,值2,值3,控件自身,值3
在控件自身这里我们触发Expanded事件,然后选中所有的复选框,奇怪的是值3这个居然会调用两次,就是因为这最后一次,把Expanded事件处理器的Selected=true又给覆盖掉了。解决这个bug倒是很容易,只需要这个事件在最后触发就行了,要么加个计数器让最后的值3不调用(不太优美),不过大家应该还记得,和LoadPostData同属一个接口的还有一个方法:RaisePostDataChangedEvent。该方法会在LoadPostData方法返回为true的时候调用,那么我们只需要将代码稍微改成这样就解决了这个bug了:
1: protected override bool LoadPostData(string postDataKey, NameValueCollection postCollection)
2: {
3: if (postCollection["data"] == "click")
4: //返回true就ok了,剩下的交给RaisePostDataChangedEvent方法吧
5: return true;
6: else
7: return base.LoadPostData(postDataKey,postCollection);
8: }
9: protected override void RaisePostDataChangedEvent()
10: {
11: OnExpanded(new EventArgs());
12: }
RaisePostDataChangedEvent方法会在所有的LoadPostData方法执行完毕后执行,所以上面的bug也不复存在了。
问题是解决了,但心里总有一个疑问,为什么最后一个复选框总会出现两次呢?这个还得从LoadPostData是谁调用的开始说起。
谁调用LoadPostData
在ASP.NET中,最后都会终结到IHttpHandler接口的ProccessRequest(HttpContext context)方法上,Page类实现了IHttpHandler接口,所以对于aspx页面来说,入口点就是那个ProccessRequest方法。查看代码不难发现最后归结到ProcessRequestMain方法上。Page的整个生命周期,以及一切的事件,比如Init啊,Load啊,什么的都是从这一条线上来的。
在这中间就调用了一个ProcessPostData方法,LoadPostData方法就是从这里调用的,那看来要查看为什么LoadPostData多调用一次的入口点就在这里了:
1: private void ProcessPostData(NameValueCollection postData, bool fBeforeLoad)
2: {
3: if (this._changedPostDataConsumers == null)
4: {
5: this._changedPostDataConsumers = new ArrayList();
6: }
7: if (postData != null)
8: {
9: foreach (string str in postData)
10: {
11: if ((str == null) || IsSystemPostField(str))
12: {
13: continue;
14: }
15: Control control = this.FindControl(str);
16: if (control == null)
17: {
18: if (fBeforeLoad)
19: {
20: if (this._leftoverPostData == null)
21: {
22: this._leftoverPostData = new NameValueCollection();
23: }
24: this._leftoverPostData.Add(str, null);
25: }
26: continue;
27: }
28: IPostBackDataHandler postBackDataHandler = control.PostBackDataHandler;
29: if (postBackDataHandler == null)
30: {
31: if (control.PostBackEventHandler != null)
32: {
33: this.RegisterRequiresRaiseEvent(control.PostBackEventHandler);
34: }
35: }
36: else
37: {
38: if ((postBackDataHandler != null) && postBackDataHandler.LoadPostData(str, this._requestValueCollection))
39: {
40: this._changedPostDataConsumers.Add(control);
41: }
42: if (this._controlsRequiringPostBack != null)
43: {
44: this._controlsRequiringPostBack.Remove(str);
45: }
46: }
47: }
48: }
49: ArrayList list = null;
50: if (this._controlsRequiringPostBack != null)
51: {
52: foreach (string str2 in this._controlsRequiringPostBack)
53: {
54: Control control2 = this.FindControl(str2);
55: if (control2 != null)
56: {
57: IPostBackDataHandler handler2 = control2._adapter as IPostBackDataHandler;
58: if (handler2 == null)
59: {
60: handler2 = control2 as IPostBackDataHandler;
61: }
62: if (handler2 == null)
63: {
64: throw new HttpException(SR.GetString("Postback_ctrl_not_found", new object[] { str2 }));
65: }
66: if (handler2.LoadPostData(str2, this._requestValueCollection))
67: {
68: this._changedPostDataConsumers.Add(control2);
69: }
70: continue;
71: }
72: if (fBeforeLoad)
73: {
74: if (list == null)
75: {
76: list = new ArrayList();
77: }
78: list.Add(str2);
79: }
80: }
81: this._controlsRequiringPostBack = list;
82: }
83: }
注意后面的foreach(string str2 in this._controlsRequiringPostBack)
这个就是多次调用LoadPostData的循环,而this._controlsRequiringPostBack又是怎么得到的呢?
如何得到_controlsRequiringPostBack
这个得看LoadAllState方法:
1: private void LoadAllState()
2: {
3: object obj2 = this.LoadPageStateFromPersistenceMedium();
4: IDictionary first = null;
5: Pair second = null;
6: Pair pair2 = obj2 as Pair;
7: if (obj2 != null)
8: {
9: first = pair2.First as IDictionary;
10: second = pair2.Second as Pair;
11: }
12: if (first != null)
13: {
14: this._controlsRequiringPostBack = (ArrayList) first["__ControlsRequirePostBackKey__"];
15: if (this._registeredControlsRequiringControlState != null)
16: {
17: foreach (Control control in (IEnumerable) this._registeredControlsRequiringControlState)
18: {
19: control.LoadControlStateInternal(first[control.UniqueID]);
20: }
21: }
22: }
23: if (second != null)
24: {
25: string s = (string) second.First;
26: int num = int.Parse(s, NumberFormatInfo.InvariantInfo);
27: this._fPageLayoutChanged = num != this.GetTypeHashCode();
28: if (!this._fPageLayoutChanged)
29: {
30: base.LoadViewStateRecursive(second.Second);
31: }
32: }
33: }
this._controlsRequiringPostBack = (ArrayList) first["__ControlsRequirePostBackKey__"];
这里的first是控件状态,而second是视图状态。
既然是控件状态,那我们去看看保存控件状态的地方。
SaveAllState方法
1: if ((this._registeredControlsThatRequirePostBack != null) && (this._registeredControlsThatRequirePostBack.Count > 0))
2: {
3: if (dictionary == null)
4: {
5: dictionary = new HybridDictionary();
6: }
7: dictionary.Add("__ControlsRequirePostBackKey__", this._registeredControlsThatRequirePostBack);
8: }
哦,原来键值为__ControlsRequirePostBackKey__的控件状态实际上就是_registeredControlsThatRequirePostBack啊,而_registeredControlsThatRequirePostBack又是怎么得来的呢?
1: [EditorBrowsable(EditorBrowsableState.Advanced)]
2: public void RegisterRequiresPostBack(Control control)
3: {
4: if (!(control is IPostBackDataHandler) && !(control._adapter is IPostBackDataHandler))
5: {
6: throw new HttpException(SR.GetString("Ctrl_not_data_handler"));
7: }
8: if (this._registeredControlsThatRequirePostBack == null)
9: {
10: this._registeredControlsThatRequirePostBack = new ArrayList();
11: }
12: this._registeredControlsThatRequirePostBack.Add(control.UniqueID);
13: }
这个方法是Page提供的一个公共方法,哪个控件想注册回发就得调用一下。一般控件都会在PreRender方法里干这个注册的事儿,那控件的PreRender方法是怎么调用的呢?
1: internal virtual void PreRenderRecursiveInternal()
2: {
3: if (!this.Visible)
4: {
5: this.flags.Set(0x10);
6: }
7: else
8: {
9: this.flags.Clear(0x10);
10: this.EnsureChildControls();
11: if (this._adapter != null)
12: {
13: this._adapter.OnPreRender(EventArgs.Empty);
14: }
15: else
16: {
17: this.OnPreRender(EventArgs.Empty);
18: }
19: if ((this._occasionalFields != null) && (this._occasionalFields.Controls != null))
20: {
21: string errorMsg = this._occasionalFields.Controls.SetCollectionReadOnly("Parent_collections_readonly");
22: int count = this._occasionalFields.Controls.Count;
23: for (int i = 0; i < count; i++)
24: {
25: this._occasionalFields.Controls[i].PreRenderRecursiveInternal();
26: }
27: this._occasionalFields.Controls.SetCollectionReadOnly(errorMsg);
28: }
29: }
30: this._controlState = ControlState.PreRendered;
31: }
上面的代码实际上就是在整个页面的控件树上递归的调用PreRenderRecursiveInternal()方法,最后调用控件树每个节点的OnPreRender方法,该方法里将有注册回发的代码。
好,这里就介绍到这里,我们再回过头来看看CheckBoxList是怎么实现的:
CheckBoxList的实现
我原本以为CheckBoxList
里会有很多CheckBox,但查看其源代码后发现只有一个,原来这里使用了原型设计模式。生成的那么多复选框都只是那一个CheckBox Render出来的。
比如在CheckBoxList的PreRender方法里注册回发的时候:
1: protected internal override void OnPreRender(EventArgs e)
2: {
3: base.OnPreRender(e);
4: this._controlToRepeat.AutoPostBack = this.AutoPostBack;
5: this._controlToRepeat.CausesValidation = this.CausesValidation;
6: this._controlToRepeat.ValidationGroup = this.ValidationGroup;
7: if (this.Page != null)
8: {
9: for (int i = 0; i < this.Items.Count; i++)
10: {
11: this._controlToRepeat.ID = i.ToString(NumberFormatInfo.InvariantInfo);
12: this.Page.RegisterRequiresPostBack(this._controlToRepeat);
13: }
14: }
15: }
上面代码中的_controlToRepeat就是那仅有的一个CheckBox。大家可以看到CheckBoxList注册回发就是遍历其Items,然后注册。
如果我们查看CheckBoxList的Controls属性发现,其Count为1,实际上就是那仅有的一个CheckBox,再回到上一节介绍的内容,在CheckBoxList的OnPreRender方法执行完毕,也就是把几个checkbox都注册回发了,然后还会调用CheckBoxList的子控件的PreRenderRecursiveInternal()方法,就是这里的仅有的那个CheckBox,经过CheckBoxList的OnPreRender方法调用后,我们发现该CheckBox的ID就等于最后一个复选框的ID,在ChexkBox的OnPreRender方法里,它又会把自己再注册一次,这样也就出现了文章开头那一幕:会对最后一个复选框调用两次LoadPostData。
疑为bug
这里的问题就是在CheckBoxList里,已经为所有的复选框注册了回发,但是因为递归调用PreRender,又会调用CheckBoxList子控件的PreRender方法,致使最后一个复选框注册两次。
后记
在这里不难发现Control的PreRenderRecursiveInternal()方法是internal virtual的,允许程序集内的子类覆盖,从Control派生的类,应该根据自己的实际情况,选择是否覆盖该方法,而这里的CheckBoxList就是其中一个,CheckBoxList无需再调用子控件的PreRender方法来再次注册会发,所以它应该覆盖Control的PreRenderRecursiveInternal方法,但是开发CheckBoxList的那位同学却没有这么干,也就造成了这种情况~
此文写的及其零乱,仅仅是个人记录。后面有时间会再仔细的加工整理一番,现在请各位看官多多见谅。