思归的“动态控件的状态问题”的分析

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

思归的“动态控件的状态问题”很有意思:

http://blog.joycode.com/saucer/archive/2004/10/20/35927.aspx

希望写下这样的logic对大家做troubleshooting有所帮助。我们可以这样分析这个问题:

第一步,简化一下Page。建一个新的ASP.Net Web Application,用下面的code:

private void Page_Load(object sender, System.EventArgs e)

{

DropDownList ddlDynamic = new DropDownList();

ddlDynamic.ID = "ddlDynamic";

HtmlForm form1 = (HtmlForm)this.FindControl("Form1");

if (!IsPostBack)

{

ddlDynamic.Items.Add("Before");

}

form1.Controls.Add(ddlDynamic);

if (!IsPostBack)

{

ddlDynamic.Items.Add("After");

}

}

在Page上扔个Button,以便可以PostBack。运行后Postback的结果,“Before” item没被保留,“After”被保留了。问题被isolate:问题不在于DropDownList或者ListCollection对于view state的处理出问题,而是特定一个ListItem view state的处理有异。

现在有目标了,接下来看ListItem source code:

internal object SaveViewState()

{

if (this.misc.Get(2) && this.misc.Get(3))

{

return new Pair(this.Text, this.Value);

}

if (this.misc.Get(2))

{

return this.Text;

}

if (this.misc.Get(3))

{

return new Pair(null, this.Value);

}

return null;

}

可以看到只有misc.Get(2)或misc.Get(3)符合一定条件才存view state,鉴于misc是private member,继续在ListItem的code里找什么会影响misc.Get(2) or misc.Get(3)的值,结果如下:

internal bool Dirty

{

set

{

this.misc.Set(2, value);

this.misc.Set(3, value);

}

}

找到了唯一的可能,在Reflector里看Set方法的Callee Graph,找到System.Web.UI.WebControls.ListItemCollection.Add(ListItem):Void方法。接续看source code:

public void Add(ListItem item)

{

this.listItems.Add(item);

if (this.marked)

{

item.Dirty = true;

}

}

这里有一个private bool marked flag。继续在ListItemCollection里找:

internal void TrackViewState()

{

this.marked = true;

for (int num1 = 0; num1 < this.Count; num1++)

{

this[num1].TrackViewState();

}

}

void IStateManager.TrackViewState()

{

this.TrackViewState();

}

好了,看来这个方法就是关键了……由于是Interface的方法,我们可以尝试一下在Page2里调用它:

((IStateManager)(ddlDynamic.Items)).TrackViewState();

if (!IsPostBack)

{

for (int i=1; i <=3; i++)

ddlDynamic.Items.Add(new ListItem(i.ToString(), i.ToString()));

}

form1.Controls.Add(ddlDynamic);

确实是起作用了……鉴于如此,我们可以猜测ControlCollection.Add一定调用了System.Web.UI.WebControls.ListItemCollection.TrackViewState()这个方法。要证明这点容易多了……

用WinDbg,在System.Web.UI.WebControls.ListItemCollection.TrackViewState()方法上设个断点。Call Stack如下:

019cf8b0 06538fd0 [DEFAULT] [hasThis] Void System.Web.UI.WebControls.ListItemCollection.TrackViewState()

019cf8b4 06538fbe [DEFAULT] [hasThis] Void System.Web.UI.WebControls.ListControl.TrackViewState()

019cf8bc 06538e53 [DEFAULT] [hasThis] Void System.Web.UI.Control.InitRecursive(Class System.Web.UI.Control)

019cf8d8 0653758a [DEFAULT] [hasThis] Void System.Web.UI.Control.AddedControl(Class System.Web.UI.Control,I4)

019cf8f4 06537462 [DEFAULT] [hasThis] Void System.Web.UI.ControlCollection.Add(Class System.Web.UI.Control)

019cf904 063c06fc [DEFAULT] [hasThis] Void WebApplication37.WebForm3.Page_Load(Object,Class System.EventArgs)

at [+0x13c] [+0x8c] c:\inetpub\wwwroot\webapplication37\webform3.aspx.cs:36

019cf944 065391a4 [DEFAULT] [hasThis] Void System.Web.UI.Control.OnLoad(Class System.EventArgs)

好了……一切都明了……

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
 
 
© 2005- 王朝網路 版權所有 導航