ASP.NET 2.0 的数据绑定函数Eval()简化掉了ASP 1.1神秘的Container.DataItem,好比数据绑定表达式: express
<%# (Container.DataItem as DataRowView)["ProductName"].ToString() %>
ASP 1.1简化为:(去掉了类型指定, Eval经过反射实现,本文再也不阐述) c#
<%# DataBinder.Eval(Container.DataItem, "ProductName").ToString() %>ASP 2.0又简化为,去掉了Container局部变量:
<%# Eval("ProductName") %>那么,Page.Eval()又是如何知道"ProductName"是那个数据的属性呢,即Container.DataItem真的消失了吗?
在Control.DataBind()中,创建,这样能够保证子控件的DataItem Container始终在栈顶。 函数
public class Control { protected virtual void DataBind(bool raiseOnDataBinding) { bool foundDataItem = false; if (this.IsBindingContainer) { object o = DataBinder.GetDataItem(this, out foundDataItem); if (foundDataItem) Page.PushDataItemContext(o); <-- 将DataItem压入堆栈 } try { if (raiseOnDataBinding) OnDataBinding(EventArgs.Empty); DataBindChildren(); <-- 绑定子控件 } finally { if (foundDataItem) Page.PopDataItemContext(); <-- 将DataItem弹出堆栈 } } }2. 获取DataItem Container
public class Page { public object GetDataItem() { ... return this._dataBindingContext.Peek(); <-- 读取堆栈顶部的DataItem Container,就是正在绑定的DataItem Container } }3. TemplateControl.Eval()
public class TemplateControl { protected string Eval (string expression, string format) { return DataBinder.Eval (Page.GetDataItem(), expression, format); } }结论: 从上面看出Page.Eval()在计算的时候仍是引用了Container.DataItem,只不过这个DataItem经过DataItem Container堆栈自动计算出来的。我认为Page.Eval()看似把问题简化了,其实把问题搞得更加神秘。