ASP.NET Eval四种绑定方式 及详解
1、1.x中的数据绑定语法
<asp:Literal id="litEval2" runat="server" Text='<%#DataBinder.Eval(Container.DataItem, "userName")%>' />
2、 2.x简化Eval数据绑定语法
<asp:Literal id="litEval1" runat="server" Text='<%Eval("userName")%>' />
3、第二种方法的方法重载
<a href='<%# Eval("userId","Default.aspx?id={0}")%>'><%# Eval("userName") %></a>
4、eval同时绑定两个值
<a href='<%# string.Format("Default.aspx?id={0}&role={1}", Eval("userId"),Eval("userRole"))%>'><%# Eval("userName") %></a>
eval_r()方法在运行时使用反射执行后期绑定计算,因此与标准的ASP.NET数据绑定方法bind相比,会导致性能明显下降。它一般用在绑定时需要格式化字符串的情况下。多数情况尽量少用此方法
Eval 方法是静态(只读)方法,该方法采用数据字段的值作为参数并将其作为字符串返回。Bind 方法支持读/写功能,可以检索数据绑定控件的值并将任何更改提交回数据库。
使用 Eval 方法
Eval 方法可计算数据绑定控件(如 GridView、DetailsView 和 FormView
控件)的模板中的后期绑定数据表达式。在运行时,Eval 方法调用 DataBinder 对象的 Eval
方法,同时引用命名容器的当前数据项。命名容器通常是包含完整记录的数据绑定控件的最小组成部分,如 GridView
控件中的一行。因此,只能对数据绑定控件的模板内的绑定使用 Eval 方法。
Eval
方法以数据字段的名称作为参数,从数据源的当前记录返回一个包含该字段值的字符串。可以提供第二个参数来指定返回字符串的格式,该参数为可选参数。字符串格式参数使用为
String 类的 Format 方法定义的语法。
使用 Bind 方法
Bind 方法与 Eval 方法有一些相似之处,但也存在很大的差异。虽然可以像使用 Eval 方法一样使用 Bind
方法来检索数据绑定字段的值,但当数据可以被修改时,还是要使用 Bind 方法。
在 ASP.NET 中,数据绑定控件(如 GridView、DetailsView 和 FormView
控件)可自动使用数据源控件的更新、删除和插入操作。例如,如果已为数据源控件定义了 SQL Select、Insert、Delete 和 Update
语句,则通过使用 GridView、DetailsView 或 FormView 控件模板中的 Bind
方法,就可以使控件从模板中的子控件中提取值,并将这些值传递给数据源控件。然后数据源控件将执行适当的数据库命令。出于这个原因,在数据绑定控件的
EditItemTemplate 或 InsertItemTemplate 中要使用 Bind 函数。
Bind 方法通常与输入控件一起使用,例如由编辑模式中的 GridView 行所呈现的 TextBox
控件。当数据绑定控件将这些输入控件作为自身呈现的一部分创建时,该方法便可提取输入值。
Bind 方法采用数据字段的名称作为参数,从而与绑定属性关联,如下面的示例所示:
<EditItemTemplate>
<table>
<tr>
<td
align=right>
<b>Employee
ID:</b>
</td>
<td>
<%# eval_r("EmployeeID")
%>
</td>
</tr>
<tr>
<td
align=right>
<b>First
Name:</b>
</td>
<td>
<asp:TextBox
ID="EditFirstNameTextBox" RunAt="Server"
Text='<%# Bind("FirstName")
%>' />
</td>
</tr>
<tr>
<td
align=right>
<b>Last
Name:</b>
</td>
<td>
<asp:TextBox
ID="EditLastNameTextBox" RunAt="Server"
Text='<%# Bind("LastName") %>'
/>
</td>
</tr>
<tr>
<td
colspan="2">
<asp:LinkButton ID="UpdateButton"
RunAt="server"
Text="Update" CommandName="Update" />
<asp:LinkButton
ID="CancelUpdateButton" RunAt="server"
Text="Cancel" CommandName="Cancel"
/>
</td>
</tr>
</table>
</EditItemTemplate>
单击行的 Update 按钮时,使用 Bind 语法绑定的每个控件属性值都会被提取出来,并传递给数据源控件以执行更新操作。
使用 DataBinder.Eval
ASP.NET 提供了一个名为 DataBinder.Eval
的静态方法,该方法计算后期绑定的数据绑定表达式,并将结果格式化为字符串(可选)。利用此方法,可以避免许多在将值强制为所需数据类型时必须执行的显式强制转换操作。
例如,在下面的代码片段中,一个整数显示为货币字符串。使用标准的 ASP.NET 数据绑定语法,必须首先强制转换数据行的类型以便检索数据字段
IntegerValue。然后,这将作为参数传递到 String.Format 方法:
<%# String.Format("{0:c}",
((DataRowView)Container.DataItem)["IntegerValue"]) %>
将此语法与 DataBinder.Eval 的语法进行比较,后者只有三个参数:数据项的命名容器、数据字段名称和格式字符串。在模板化列表中(如
DataList 类、DataGrid 类或 Repeater 类),命名容器始终是 Container.DataItem。
<%# DataBinder.eval_r(Container.DataItem, "IntegerValue", "{0:c}") %>
格式字符串参数是可选的。如果它被忽略,DataBinder.Eval 将返回类型对象的值,如下面的示例所示:
<%#
(bool)DataBinder.eval_r(Container.DataItem, "BoolValue")
%>
当对模板化列表中的控件进行数据绑定时,DataBinder.Eval 特别有用,因为数据行和数据字段通常都必须强制转换。
在TemplateField模板中为了能够有限制的或者取出数据库中某列的值时,可以用Bind和Eval方法来实现。以下是Bind方法的格式,Eval的格式也是和Bind一样的。
Bind("列的名称","显示的格式文")
比如我们要取个日期型的数据,在数据库中列名是updated,数值是2008/06/01。但是想2008年06月01日这样显示,我们可以这样来写Bind("updated",
"{0:yyyy年MM月dd日}"),Eval也是如此。
两者都能读取数据中的值,并显示。当我们使用编辑更新操作时,Bind能够自动的将修改的值更新到数据库中,并显示出修改后的值。但是用了Eval却只能得到错误画面,新的数据没有更新到数据库中。
从这点看来,Bind方法和Eval方法的区别就是:Bind方法在读取和更新数据这2方面都是可以,但是Eval方法只能读取显示数据。所以,我们在选择Bind方法和Eval方法的时候,必须要有争对性,当数据肯定需要更新操作的时候我们应该使用Bind,只是显示数据,不会有任何操作的就可以使用Eval方法。
在更新操作中我们可以在GridView1_RowUpdating事件中操作,例子如下:
如果我们能充分理解Bind方法和Eval方法,其实也就没必要向上面那样去写,都是可以自动完成的。上面的方法除了比较复杂的操作才会用到,这也是一个使用技巧。
ASP.NET中的Bind和Eval的区别详解
eval_r()方法在运行时使用反射执行后期绑定计算,因此与标准的ASP.NET数据绑定方法bind相比,会导致性能明显下降。它一般用在绑定时需要格式化字符串的情况下。多数情况尽量少用此方法
Eval 方法是静态(只读)方法,该方法采用数据字段的值作为参数并将其作为字符串返回。Bind 方法支持读/写功能,可以检索数据绑定控件的值并将任何更改提交回数据库。
使用DataBinder.Eval
ASP.NET 提供了一个名为 DataBinder.Eval 的静态方法,该方法计算后期绑定的数据绑定表达式,并将结果格式化为字符串(可选)。利用此方法,可以避免许多在将值强制为所需数据类型时必须执行的显式强制转换操作。
例如,在下面的代码片段中,一个整数显示为货币字符串。使用标准的ASP.NET 数据绑定语法,必须首先强制转换数据行的类型以便检索数据字段IntegerValue。然后,这将作为参数传递到String.Format 方法:
<%#
String.Format("{0:c}", ((DataRowView)Container.DataItem)["IntegerValue"])
%>
将此语法与DataBinder.Eval 的语法进行比较,后者只有三个参数:数据项的命名容器、数据字段名称和格式字符串。在模板化列表中(如DataList 类、DataGrid 类或Repeater 类),命名容器始终是Container.DataItem。
<%# DataBinder.eval_r(Container.DataItem,
"IntegerValue", "{0:c}") %>
也可以同时绑定实体类
<%# DataBinder.Eval(Container.DataItem, "Item.Product.Id") %>
其中的数据源为IList<LineItem>,各个类的关系为:
格式字符串参数是可选的。如果它被忽略,DataBinder.Eval 将返回类型对象的值,如下面的示例所示:
<%#
(bool)DataBinder.eval_r(Container.DataItem, "BoolValue") %>
当对模板化列表中的控件进行数据绑定时,DataBinder.Eval 特别有用,因为数据行和数据字段通常都必须强制转换。
Eval和Bind的格式化字符串所处位置不同,以下举出几个例子:
1.Eval是单向取值,既可以使用系统定义的函数格式化,也可以调用自己定义的格式化函数(如例子中的MyFormatFunction函数)。
<%#
string.Format("{0:yyyy-MM-dd}",eval_r("BirthDate")) %>
<%#
MyFormatFunction(eval_r("BirthDate")) %>
2.Bind是双向的,格式化字符串既被用于格式化显示,又要作为数据输入验证。
<%# Bind("BirthDate","{0:yyyy-MM-dd}")
%>
<%# Bind("Birthday","{0:M.d}") %>
DataFormatString="{0:格式字符串}"
1、时间格式化:{0:yyyy-MM-dd}
//绑定日期字段格式字符串
<%# Convert.ToDateTime(((System.Data.DataRowView)Container.DataItem)["FbTime"]).ToString("yyyy年MM月dd日") %>
<%# DataBinder.eval_r(Container, "DataItem.FbTime","{0:yyyy年MM月dd日}") %>
//字符串绑定超过指定长度截断
<%# DataBinder.eval_r(Container.DataItem, "Text").ToString().Trim().Length>7?DataBinder.eval_r(Container.DataItem, "Text").ToString().Trim().Substring(0,7):DataBinder.eval_r(Container.DataItem, "Text").ToString().Trim() %>
2、格式数字字符串
输入
结果
"{0:C}" 12345.6789 $12,345.68
"{0:C}" -12345.6789
($12,345.68)
"{0:D}" 12345 12345
"{0:D8}" 12345 00012345
"{0:E}"
12345.6789 1234568E+004
"{0:E10}" 12345.6789 1.2345678900E+004
"{0:F}"
12345.6789 12345.68
"{0:F0}" 12345.6789 12346
"{0:G}" 12345.6789
12345.6789
"{0:G7}" 123456789 1.234568E8
"{0:N}" 12345.6789
12,345.68
"{0:N4}" 123456789 123,456,789.0000
"Total: {0:C}" 12345.6789
Total: $12345.68
DataFormatString="{0:P}" //自动转换为百分数
C 以货币格式显示数值。0
12345.ToString("C"); //生成 ¥12,345.00
D 以十进制格式显示数值。
E 以科学记数法(指数)格式显示数值。
//价格格试化为2位小数
<%# Convert.ToDecimal(((System.Data.DataRowView)Container.DataItem)
["Price"]).ToString("F2") %>
<%# DataBinder.eval_r(Container, "DataItem.Price","{0:F2}") %>
F 以固定格式显示数值。
G 以常规格式显示数值。
N 以数字格式显示数值。
X 以十六进制格式显示数值。
12345.ToString("p"); //生成 1,234,500.00%
Eval内部机制
本文假设你已经了解ASP.NET 1.1的数据绑定(特别是Container这个局部变量)的机制,这里主要分析ASP 2.0数据绑定做了那些改进。
ASP.NET 2.0 的数据绑定函数Eval()简化掉了ASP
1.1神秘的Container.DataItem,比如数据绑定表达式:
<%# (Container.DataItem as DataRowView)["ProductName"].ToString() %> |
ASP.NET 1.1简化为:(去掉了类型指定,
Eval通过反射实现,本文不再阐述)
<%# DataBinder.Eval(Container.DataItem, "ProductName").ToString() %> |
ASP.NET 2.0又简化为,去掉了Container局部变量:
<%# Eval("ProductName") %> |
那么,Page.Eval()又是如何知道"ProductName"是那个数据的属性呢,即Container.DataItem真的消失了吗?
Eval()是Page的父类TemplateControl的方法
TemplateControl.Eval()可以自动计算出Container,
机制就是从一个dataBindingContext:Stack堆栈来获取。
1. 建立DataItem Container
栈:
在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(); |
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()看似把问题简化了,其实把问题搞得更加神秘。
ASP.NET Eval四种绑定方式 及详解的更多相关文章
- ASP.NET Eval四种绑定方式
1.1.x中的数据绑定语法 <asp:Literal id="litEval2" runat="server" Text='<%#DataBinde ...
- Spring AOP四种实现方式Demo详解与相关知识探究
一.前言 在网络上看到一篇博客Spring实现AOP的4种方式,博主写的很通俗易懂,但排版实在抓狂,对于我这么一个对排版.代码格式有强迫症的人来说,实在是不能忍受~~~~(>_<)~~~~ ...
- this四种绑定方式之间的奇淫技巧
写在前面 上一篇中,我们对于JavaScript中原始值.复杂值以及内存空间进行了一个深入浅出的总结,这次我们来聊一聊JavaScript中this关键字的深入浅出的用法. 在 JavaScript ...
- JDK提供的四种线程池代码详解
一.线程池什么时候使用,会给我们带来什么好处? 如果很多用户去访问服务器,用户访问服务器的时间是非常短暂的,那么有可能在创建线程和销毁线程上花费的时间会远远大于访问所消耗的时间,如果采用线程池会使线程 ...
- EF三种编程方式图文详解
Entity Framework4.1之前EF支持“Database First”和“Model First”编程方式,从EF4.1开始EF开始支持支持“Code First”编程方式,今天简单看一下 ...
- C++中四种强制类型转换区别详解
C++即支持C风格的类型转换,又有自己风格的类型转换.C风格的转换格式很简单,但是有不少缺点的: 1.转换太过随意,可以在任意类型之间转换.你可以把一个指向const对象的指针转换成指向非const对 ...
- C++中的四种强制类型转换符详解
阅读目录 C++即支持C风格的类型转换,又有自己风格的类型转换.C风格的转换格式很简单,但是有不少缺点的: 转换太过随意,可以在任意类型之间转换.你可以把一个指向const对象的指针转换成指向非con ...
- MySQL 四种事务隔离级别详解及对比--转
http://www.jb51.net/article/100183.htm 接的隔离级别.它的语法如下: ? 1 SET [SESSION | GLOBAL] TRANSACTION ISOLATI ...
- MySQL四种事务隔离级别详解
本文实验的测试环境:Windows 10+cmd+MySQL5.6.36+InnoDB 一.事务的基本要素(ACID) 1.原子性(Atomicity):事务开始后所有操作,要么全部做完,要么全部不做 ...
随机推荐
- Webx框架:依赖注入
Webx的依赖注入和Spring的依赖注入很像,仅仅是有一点点的差别. 注入的时候仅仅能让生命周期长的注入到生命周期短的对象中,比方requestScope对象注入到singleton时就会错误发生. ...
- VS类添加头文件注释
VS2015参考: http://blog.csdn.net/qq395537505/article/details/50853546 修改两个文件,详细信息 VS2010: 找到VS的安装目录 E ...
- 网络驱动移植之例解netdev_priv函数
版权声明:本文为博主原创文章,未经博主允许不得转载. 开发平台:Ubuntu 11.04 编译器:gcc version 4.5.2 (Ubuntu/Linaro 4.5.2-8ubuntu4) 内核 ...
- POJ3020 Antenna Placement —— 最大匹配 or 最小边覆盖
题目链接:https://vjudge.net/problem/POJ-3020 Antenna Placement Time Limit: 1000MS Memory Limit: 65536K ...
- validationEngine验证的使用
改校验的方法功能很强大,具体查看api http://code.ciaoca.com/jquery/validation_engine/ 效果:
- BZOJ_3620_似乎在梦中见过的样子_KMP
BZOJ_3620_似乎在梦中见过的样子_KMP Description “Madoka,不要相信 QB!”伴随着 Homura 的失望地喊叫,Madoka 与 QB 签订了契约. 这是 Modoka ...
- artemplate include
include用于嵌入字模板 {{include 'template_name'}} 子模板 默认共享当前的数据 也可以自己指定数据 {{include 'template_name' templat ...
- .NET MVC API返回JSON对象
方法多种,自己目前采用的是自定义返回格式的方法,不需要修改配置文件. 辅助类: public class ApiResponseHelper { public static HttpResponseM ...
- ChartCtrl源码剖析之——CChartAxis类
CChartAxis类用来绘制波形控件的坐标轴,这个源码相对较复杂,当初阅读的时候耗费了不少精力来理解源码中的一些实现细节. CChartAxis类的头文件. #if !defined(AFX_CHA ...
- 利用HashMap存取对象并获得键值集合
1.HashMap 已实现的接口 Serializable, Cloneable, Map<K,V> 2.方法摘要 相关代码 /** * * @param ha * write(HashM ...