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()看似把問題簡化了,其實把問題搞得更加神祕。