Eval()和DataBinder.Eval()方法..
时间:2010-09-27 来源:张静静
ASP.NET中的Eval()和DataBinder.Eval()方法时间:
Eval( " ")和Bind( " ") 这两种一个单向绑定,一个
双向绑定
Eval( " ")和Bind( " ") 这两种一个单向绑定,一个双向绑定
bind是双向绑定,但需数据源可更改才能用
ASP.NET 2.0改善了模板中的数据绑定操作,把v1.x中的数据绑定语法
DataBinder.Eval(Container.DataItem, fieldname)简化为Eval(fieldname)
。Eval方法与DataBinder.Eval一样可以接受一个可选的格式化字符串参数。
缩短的Eval语法与DataBinder.Eval的不同点在于,Eval会根据最近的容器对
象(例如DataListItem)的DataItem属性来自动地解析字段,而
DataBinder.Eval需要使用参数来指定容器。由于这个原因,Eval只能在数据
绑定控件的模板中使用,而不能用于Page(页面)层。当然,ASP.NET 2.0页
面中仍然支持DataBinder.Eval,你可以在不支持简化的Eval语法的环境中使
用它。
下面的例子演示了如何使用新的简化的Eval数据绑定语法绑定到DataList数据
项模板(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 ">
数据绑定也可以作为控件的主题定义(theme definition)的一部分,这
样我们就可以通过改变主题来随意地改变模板化控件的布局和外观。但是
Theme(主题)模板中只能使用Eval(或者后面讨论的Bind)。绑定到任意的
用户代码是被禁止的。
Q:DataBinder.Eval(Container.DataItem,"Name")和Container.DataItem
("Name")有什么区别?
A:DataBinder是System.Web里面的一个静态类,它提供了Eval方法用于简化数
据绑定表达式的编写,但是它使用的方式是通过Reflection等开销比较大的方
法来达到易用性,因此其性能并不是最好的。而Container则根本不是任何一
个静态的对象或方法,它是ASP.NET页面编译器在数据绑定事件处理程序内部
声明的局部变量,其类型是可以进行数据绑定的控件的数据容器类型(如在
Repeater内部的数据绑定容器叫RepeaterItem),在这些容器类中基本都有
DataItem属性,因此你可以写Container.DataItem,这个属性返回的是你正在
被绑定的数据源中的那个数据项。如果你的数据源是DataTable,则这个数据
项的类型实际是DataRowView。
Q:Eval()方法的源代码?DataBinder.Eval()和Eval()各个用法的性能区别
?
A:实际上Eval方法是TemplateControl的,而System.Web.UI.Page和
System.Web.UI.UserControl都继承于TemplateControl,所以我们可以在Page
和UserControl上直接调用个方法。
Page.Eval方法可以帮助我们更好的撰写数据绑定表达式,在ASP.NET 1.x时代
,数据绑定表达式的一般形式是:
<%# DataBinder.Eval( Container , “DataItem.Name”) %>
而在ASP.NET 2.0中,同样的代码,我们可以这样写:
<%# Eval( “Name” )%>
ASP.NET 2.0是怎么实现的呢?我们先从Eval方法来研究,通过反射.NET
Framework 2.0类库的源代码,我们可以看到这个方法是这样实现的:
protected internal object Eval(string expression)
{
this.CheckPageExists();
return DataBinder.Eval(this.Page.GetDataItem(), expression);
}
第一行我们不必管,这是检查调用的时候有没有Page对象的,如果没有则会抛
出一个异常。
关键是第二行:
return DataBinder.Eval(this.Page.GetDataItem(), expression);
Page.GetDataItem()也是2.0中新增的一个方法,用途是正是取代ASP.NET 1.x
中的Container.DataItem。
看来不摸清楚GetDataItem()方法,我们也很难明白Eval的原理。GetDataItem
的实现也很简单:
public object GetDataItem()
{
if ((this._dataBindingContext == null) ||
(this._dataBindingContext.Count == 0))
{
throw new InvalidOperationException(SR.GetString
("Page_MissingDataBindingContext"));
}
return this._dataBindingContext.Peek();
}
我们注意到了有一个内部对象_dataBindingContext,通过查源代码发现这是
一个Stack类型的东西。所以他有Peek方法。而这一段代码很容易看懂,先判
断这个Stack是否被实例化,然后,判断这个Stack里面是不是有任何元素,如
果Stack没有被实例化或者没有元素则抛出一个异常。最后是将这个堆栈顶部
的元素返回。
ASP.NET 2.0用了一个Stack来保存所谓的DataItem,我们很快就查到了为这个
堆栈压元素和弹出元素的方法:Control.DataBind方法:
protected virtual void DataBind(bool raiseOnDataBinding)
{
bool flag1 = false;//这个标志的用处在上下文中很容易推出来,如果有
DataItem压栈,则在后面出栈。
if (this.IsBindingContainer)//判断控件是不是数据绑定容器,实际上就是
判断控件类是不是实现了INamingContainer
{
bool flag2;
object obj1 = DataBinder.GetDataItem(this, out flag2);//这个方法是判
断控件是不是有DataItem属性,并把它取出来。
if (flag2 && (this.Page != null))//如果控件有DataItem
{
this.Page.PushDataBindingContext(obj1);//把DataItem压栈,
PushDataBindingContext就是调用_dataBindingContext的Push方法
flag1 = true;
}
}
try
{
if (raiseOnDataBinding)//这里是判断是不是触发DataBinding事件的。
{
this.OnDataBinding(EventArgs.Empty);
}
this.DataBindChildren();//对子控件进行数据绑定,如果这个控件有
DataItem,则上面会将DataItem压入栈顶,这样,在子控件里面调用Eval或者
GetDataItem方法,就会把刚刚压进去的DataItem给取出来。
}
finally
{
if (flag1)//如果刚才有压栈,则现在弹出来。
{
this.Page.PopDataBindingContext();//PopDataBindingContext就是调用
_dataBindingContext的Pop方法
}
}
}
至此,我们已经可以完全了解ASP.NET 2.0中GetDataIten和Eval方法运作的原
理了
-常见绑定格式,不过他们的性能有区别。
<%# 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()了)
上面这三个性能最好。
下面为使用Eval()方法的一个小例子:
以下为引用的内容:
<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 ">