ASP.NET中的Eval()和DataBinder.Eval()方法
Eval( " ")和Bind( " ") 這兩種一個單向綁定,一個雙向綁定
bind是雙向綁定,但需數(shù)據(jù)源可更改才能用
ASP.NET 2.0改善了模板中的數(shù)據(jù)綁定操作,把v1.x中的數(shù)據(jù)綁定語法DataBinder.Eval(Container.DataItem, fieldname)簡化為Eval(fieldname)。Eval方法與DataBinder.Eval一樣可以接受一個可選的格式化字符串參數(shù)。縮短的Eval語法與DataBinder.Eval的不同點在于,Eval會根據(jù)最近的容器對象(例如DataListItem)的DataItem屬性來自動地解析字段,而DataBinder.Eval需要使用參數(shù)來指定容器。由于這個原因,Eval只能在數(shù)據(jù)綁定控件的模板中使用,而不能用于Page(頁面)層。當(dāng)然,ASP.NET 2.0頁面中仍然支持DataBinder.Eval,你可以在不支持簡化的Eval語法的環(huán)境中使用它。
下面的例子演示了如何使用新的簡化的Eval數(shù)據(jù)綁定語法綁定到DataList數(shù)據(jù)項模板(ItemTemplate)中的Image、Label和HyperLink控件。
<asp:DataList ID= "DataList1 " RepeatColumns= "5 " Width= "600 " runat= "server " DataSourceID= "ObjectDataSource1 ">
<ItemTemplate>
<asp:HyperLink ID= "HyperLink1 " runat= "server " NavigateUrl= '<%# Eval( "PhotoID ", "PhotoFormViewPlain.aspx?ID={0} ") %> '>
<asp:Image ID= "Image1 " Runat= "server " ImageUrl= '<%# Eval( "FileName ", "images/thumbs/{0} ") %> ' /></asp:HyperLink>
<asp:Label ID= "CaptionLabel " runat= "server " Text= '<%# Eval( "Caption ") %> ' />
</ItemTemplate>
</asp:DataList><br />
<asp:ObjectDataSource ID= "ObjectDataSource1 " runat= "server " TypeName= "DataComponentTableAdapters.PhotosTableAdapter " SelectMethod= "GetPhotosForAlbum ">
數(shù)據(jù)綁定也可以作為控件的主題定義(theme definition)的一部分,這樣我們就可以通過改變主題來隨意地改變模板化控件的布局和外觀。但是Theme(主題)模板中只能使用Eval(或者后面討論的Bind)。綁定到任意的用戶代碼是被禁止的。
Q:DataBinder.Eval(Container.DataItem,"Name")和Container.DataItem("Name")有什么區(qū)別?
A:DataBinder是System.Web里面的一個靜態(tài)類,它提供了Eval方法用于簡化數(shù)據(jù)綁定表達(dá)式的編寫,但是它使用的方式是通過Reflection等開銷比較大的方法來達(dá)到易用性,因此其性能并不是最好的。而Container則根本不是任何一個靜態(tài)的對象或方法,它是ASP.NET頁面編譯器在數(shù)據(jù)綁定事件處理程序內(nèi)部聲明的局部變量,其類型是可以進(jìn)行數(shù)據(jù)綁定的控件的數(shù)據(jù)容器類型(如在Repeater內(nèi)部的數(shù)據(jù)綁定容器叫RepeaterItem),在這些容器類中基本都有DataItem屬性,因此你可以寫Container.DataItem,這個屬性返回的是你正在被綁定的數(shù)據(jù)源中的那個數(shù)據(jù)項。如果你的數(shù)據(jù)源是DataTable,則這個數(shù)據(jù)項的類型實際是DataRowView。
Q:Eval()方法的源代碼?DataBinder.Eval()和Eval()各個用法的性能區(qū)別?
A:實際上Eval方法是TemplateControl的,而System.Web.UI.Page和System.Web.UI.UserControl都繼承于TemplateControl,所以我們可以在Page和UserControl上直接調(diào)用個方法。
Page.Eval方法可以幫助我們更好的撰寫數(shù)據(jù)綁定表達(dá)式,在ASP.NET 1.x時代,數(shù)據(jù)綁定表達(dá)式的一般形式是:
<%# DataBinder.Eval( Container , “DataItem.Name”) %>
而在ASP.NET 2.0中,同樣的代碼,我們可以這樣寫:
<%# Eval( “Name” )%>
ASP.NET 2.0是怎么實現(xiàn)的呢?我們先從Eval方法來研究,通過反射.NET Framework 2.0類庫的源代碼,我們可以看到這個方法是這樣實現(xiàn)的:
protected internal object Eval(string expression)
{
this.CheckPageExists();
return DataBinder.Eval(this.Page.GetDataItem(), expression);
}
第一行我們不必管,這是檢查調(diào)用的時候有沒有Page對象的,如果沒有則會拋出一個異常。
關(guān)鍵是第二行:
return DataBinder.Eval(this.Page.GetDataItem(), expression);
Page.GetDataItem()也是2.0中新增的一個方法,用途是正是取代ASP.NET 1.x中的Container.DataItem。
看來不摸清楚GetDataItem()方法,我們也很難明白Eval的原理。GetDataItem的實現(xiàn)也很簡單:
public object GetDataItem()
{
if ((this._dataBindingContext == null) || (this._dataBindingContext.Count == 0))
{
throw new InvalidOperationException(SR.GetString("Page_MissingDataBindingContext"));
}
return this._dataBindingContext.Peek();
}
我們注意到了有一個內(nèi)部對象_dataBindingContext,通過查源代碼發(fā)現(xiàn)這是一個Stack類型的東西。所以他有Peek方法。而這一段代碼很容易看懂,先判斷這個Stack是否被實例化,然后,判斷這個Stack里面是不是有任何元素,如果Stack沒有被實例化或者沒有元素則拋出一個異常。最后是將這個堆棧頂部的元素返回。
ASP.NET 2.0用了一個Stack來保存所謂的DataItem,我們很快就查到了為這個堆棧壓元素和彈出元素的方法:Control.DataBind方法:
protected virtual void DataBind(bool raiseOnDataBinding)
{
bool flag1 = false;//這個標(biāo)志的用處在上下文中很容易推出來,如果有DataItem壓棧,則在后面出棧。
if (this.IsBindingContainer)//判斷控件是不是數(shù)據(jù)綁定容器,實際上就是判斷控件類是不是實現(xiàn)了INamingContainer
{
bool flag2;
object obj1 = DataBinder.GetDataItem(this, out flag2);//這個方法是判斷控件是不是有DataItem屬性,并把它取出來。
if (flag2 && (this.Page != null))//如果控件有DataItem
{
this.Page.PushDataBindingContext(obj1);//把DataItem壓棧,PushDataBindingContext就是調(diào)用_dataBindingContext的Push方法
flag1 = true;
}
}
try
{
if (raiseOnDataBinding)//這里是判斷是不是觸發(fā)DataBinding事件的。
{
this.OnDataBinding(EventArgs.Empty);
}
this.DataBindChildren();//對子控件進(jìn)行數(shù)據(jù)綁定,如果這個控件有DataItem,則上面會將DataItem壓入棧頂,這樣,在子控件里面調(diào)用Eval或者GetDataItem方法,就會把剛剛壓進(jìn)去的DataItem給取出來。
}
finally
{
if (flag1)//如果剛才有壓棧,則現(xiàn)在彈出來。
{
this.Page.PopDataBindingContext();//PopDataBindingContext就是調(diào)用_dataBindingContext的Pop方法
}
}
}
至此,我們已經(jīng)可以完全了解ASP.NET 2.0中GetDataIten和Eval方法運(yùn)作的原理了
-
常見綁定格式,不過他們的性能有區(qū)別。
-
<%# DataBinder.Eval(Container.DataItem, "[n]") %>
<%# DataBinder.Eval(Container.DataItem, "ColumnName") %>
<%# DataBinder.Eval(Container.DataItem, "ColumnName", null) %>
<%# DataBinder.Eval(Container, "DataItem.ColumnName", null) %>
<%# ((DataRowView)Container.DataItem)["ColumnName"] %>
<%# ((DataRowView)Container.DataItem).Row["ColumnName"] %>
<%# ((DataRowView)Container.DataItem)["adtitle"] %>
<%# ((DataRowView)Container.DataItem)[n] %>
<%# ((DbDataRecord)Container.DataItem)[0] %>
<%# (((自定義類型)Container.DataItem)).屬性.ToString() %>(如果屬性為字符串類型就不用ToString()了)
上面這三個性能最好。
轉(zhuǎn)載于:https://www.cnblogs.com/runy/archive/2009/10/13/1582574.html
總結(jié)
以上是生活随笔為你收集整理的ASP.NET中的Eval()和DataBinder.Eval()方法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 莫烦nlp-GPT 单向语言模型
- 下一篇: 设计模式(6)六大原则之开闭原则