|
在ASP.NET我們在使用Repeater,DetailsView,FormView,GridView等數據綁定模板時,都會使用<%# Eval("字段名") %>或<%# Bind("字段名") %>這樣的語法來單向或雙向綁定數據。但是我們卻很少去了解,在這些語法的背后,ASP.NET究竟都做了哪些事情來方便我們使用這樣的語法來綁定數據。究竟解析這樣的語法是在編譯時,還是運行時?如果沒有深入去了解,我們肯定不得而知。這個簡短的系列文章就是帶我們大家一起去深入探究一下ASP.NET綁定語法的內部機理,以讓我們更加全面的認識和運用它。
事件的起因是,我希望動態的為Repeater控件添加行項模板,我可以通過實現ITempate接口的方式來動態添加行模板。并希望它通過普通的頁面綁定語法來完成數據字段的綁定功能,如下就是一個簡單的例子:
/// <summary>
/// Summary description for DynamicTemplate
/// </summary>
public class DynamicTemplate : ITemplate
{
public DynamicTemplate()
{
//
// TODO: Add constructor logic here
//
}
#region ITemplate Members
public void InstantiateIn(Control container)
{
TextBox textBox = new TextBox();
textBox.Text = @"<%# Eval(""ID"") %>";
container.Controls.Add(textBox);
}
#endregion
}在.NET 2.0中新增了雙向的數據綁定方式,主要用在GridView,DetailsView,FormView等數據容器控件中,結合DataSourceControl就可以非常輕松的完成數據的更新和提交工作,而不需要我們手工去遍歷輸入控件的值。那在這樣的雙向數據綁定中,ASP.NET又是做了哪些工作,來為我們透明輸入控件與字段的取值與對應關系,讓我們可以在DataSouceControl中方便得到數據項修改前的值和修改后的值?下面就讓我們一起來從一段頁面代碼開始吧:
<ASP:DetailsDataSouce ID="DetailsDataSouce1" runat="server">
</ASP:DetailsDataSouce>
<ASP:DetailsView ID="detailsView" runat="server" DefaultMode="Edit" DataSourceID="DetailsDataSouce1">
<Fields>
<ASP:TemplateField>
<HeaderTemplate>
電流:</HeaderTemplate>
<EditItemTemplate>
<ASP:TextBox ID="textBox1" runat="server" Text='<%# Bind("[電流{a}]")%>'></ASP:TextBox>
</EditItemTemplate>
</ASP:TemplateField>
</Fields>
</ASP:DetailsView>在了解了數據綁定語法的原理后,我還想來談談我中實踐過程中遇到的一些問題以及其它實用的綁定技巧。首先我們就來說說,特殊字段名的問題。我們知道在數據庫當中,如果表名或字段名中包含有一些特殊的不能是合法的字符時,都會使用[]將它們引起來,以便他們能夠正常使用。但是在<%# Eval("")%>的綁定語句當中,同時可以使用[],但是對于字段名中包含"(",")","[","]"這4個字符卻始終運行出錯。假設像我下面這樣來綁定"電壓(V)":<%# Eval("[電壓(V)]")%>
那么就會得到一個運行時錯誤:
DataBinding:“System.Data.DataRowView”不包含名為“電壓”的屬性。
表明括號是被認為是一個特殊字符,那我們如果給字段名加上[],如下:<%# Eval("[電壓(V)]")%>
此時,我們會得到另一個運行時錯誤:
電壓(V 既不是表 DataTable1 的 DataColumn 也不是 DataRelation。
表明,即使加上[]也無法解決這個特殊字段名的問題。同時字段名中如果也存在中括號,也是會出現這樣的問題的。但是這樣的字段名卻在GridView的自動生成列中能被正常綁定呢?問題會出現在哪里呢?分析和對比GridView的自動生成列與Eval這樣的綁定語法在最終執行綁定代碼上的不同,我們可以發現,GridView的自動生成列取值并不是使用DataBinder.Eval這個方法,它內部有自己的取值方式,但是在實現上卻是大同小異的。那究竟是在哪里出現了問題呢?我們找出DataBinder類的定義:
代碼
[ASPNETHostingPermission(SecurityAction.LinkDemand, Level=200)]
public sealed class DataBinder
{
// Fields
private static readonly char[] expressionPartSeparator = new char[] { '.' };
private static readonly char[] indexExprEndChars = new char[] { ']', ')' };
private static readonly char[] indexExprStartChars = new char[] { '[', '(' };
// Methods
public static object Eval(object container, string expression)
{
if (expression == null)
{
throw new ArgumentNullException("expression");
}
expression = expression.Trim();
if (expression.Length == 0)
{
throw new ArgumentNullException("expression");
}
if (container == null)
{
return null;
}
string[] expressionParts = expression.Split(expressionPartSeparator);
return Eval(container, expressionParts);
}
private static object Eval(object container, string[] expressionParts)
{
object propertyValue = container;
for (int i = 0; (i < expressionParts.Length)&& (propertyValue != null); i++)
{
string propName = expressionParts[i];
if (propName.IndexOfAny(indexExprStartChars)< 0)
{
propertyValue = GetPropertyValue(propertyValue, propName);
}
else
{
propertyValue = GetIndexedPropertyValue(propertyValue, propName);
}
}
return propertyValue;
}
public static string Eval(object container, string expression, string format)
{
object obj2 = Eval(container, expression);
if ((obj2 == null)||(obj2 == DBNull.Value))
{
return string.Empty;
}
if (string.IsNullOrEmpty(format))
{
return obj2.ToString();
}
return string.Format(format, obj2);
}
public static object GetDataItem(object container)
{
bool flag;
return GetDataItem(container, out flag);
}
public static object GetDataItem(object container, out bool foundDataItem)
{
if (container == null)
{
foundDataItem = false;
return null;
}
IDataItemContainer container2 = container as IDataItemContainer;
if (container2 != null)
{
foundDataItem = true;
return container2.DataItem;
}
string name = "DataItem";
PropertyInfo property = container.GetType().GetProperty(name, BindingFlags.Public | BindingFlags.Instance | BindingFlags.IgnoreCase);
if (property == null)
{
foundDataItem = false;
return null;
}
foundDataItem = true;
return property.GetValue(container, null);
}
public static object GetIndexedPropertyValue(object container, string expr)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
if (string.IsNullOrEmpty(expr))
{
throw new ArgumentNullException("expr");
}
object obj2 = null;
bool flag = false;
int length = expr.IndexOfAny(indexExprStartChars);
int num2 = expr.IndexOfAny(indexExprEndChars, length + 1);
if (((length < 0)||(num2 < 0))||(num2 == (length + 1)))
{
throw new ArgumentException(SR.GetString("DataBinder_Invalid_Indexed_Expr", new object[] { expr }));
}
string propName = null;
object obj3 = null;
111: string s = expr.Substring(length + 1,(num2 - length)- 1).Trim();
if (length != 0)
{
propName = expr.Substring(0, length);
}
if (s.Length != 0)
{
if (((s[0] == '"')&& (s[s.Length - 1] == '"'))||((s[0] == '/'')&& (s[s.Length - 1] == '/'')))
{
obj3 = s.Substring(1, s.Length - 2);
}
else if (char.IsDigit(s[0]))
{
int num3;
flag = int.TryParse(s, NumberStyles.Integer, CultureInfo.InvariantCulture, out num3);
if (flag)
{
obj3 = num3;
}
else
{
obj3 = s;
}
}
else
{
obj3 = s;
}
}
if (obj3 == null)
{
throw new ArgumentException(SR.GetString("DataBinder_Invalid_Indexed_Expr", new object[] { expr }));
}
object propertyValue = null;
if ((propName != null)&& (propName.Length != 0))
{
propertyValue = GetPropertyValue(container, propName);
}
else
{
propertyValue = container;
}
if (propertyValue == null)
{
return obj2;
}
Array array = propertyValue as Array;
if ((array!= null)&& flag)
{
return array.GetValue((int) obj3);
}
if ((propertyValue is IList)&& flag)
{
return ((IList) propertyValue)[(int) obj3];
}
PropertyInfo info = propertyValue.GetType().GetProperty("Item", BindingFlags.Public | BindingFlags.Instance, null, null, new Type[] { obj3.GetType()}, null);
if (info == null)
{
throw new ArgumentException(SR.GetString("DataBinder_No_Indexed_Accessor", new object[] { propertyValue.GetType().FullName }));
}
return info.GetValue(propertyValue, new object[] { obj3 });
}
public static string GetIndexedPropertyValue(object container, string propName, string format)
{
object indexedPropertyValue = GetIndexedPropertyValue(container, propName);
if ((indexedPropertyValue == null)||(indexedPropertyValue == DBNull.Value))
{
return string.Empty;
}
if (string.IsNullOrEmpty(format))
{
return indexedPropertyValue.ToString();
}
return string.Format(format, indexedPropertyValue);
}
public static object GetPropertyValue(object container, string propName)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
if (string.IsNullOrEmpty(propName))
{
throw new ArgumentNullException("propName");
}
PropertyDescriptor descriptor = TypeDescriptor.GetProperties(container).Find(propName, true);
if (descriptor == null)
{
throw new HttpException(SR.GetString("DataBinder_Prop_Not_Found", new object[] { container.GetType().FullName, propName }));
}
return descriptor.GetValue(container);
}
public static string GetPropertyValue(object container, string propName, string format)
{
object propertyValue = GetPropertyValue(container, propName);
if ((propertyValue == null)||(propertyValue == DBNull.Value))
{
return string.Empty;
}
if (string.IsNullOrEmpty(format))
{
return propertyValue.ToString();
}
return string.Format(format, propertyValue);
}
internal static bool IsNull(object value)
{
if ((value != null)&& !Convert.IsDBNull(value))
{
return false;
}
return true;
}
}
NET技術:深入ASP.NET數據綁定,轉載需保留來源!
鄭重聲明:本文版權歸原作者所有,轉載文章僅為傳播更多信息之目的,如作者信息標記有誤,請第一時間聯系我們修改或刪除,多謝。