揭秘ASP.NET 2.0之Page.Eval

 

实际上Eval方法是TemplateControl的,而System.Web.UI.Page和System.Web.UI.UserControl都继承于TemplateControl,所以我们可以在Page和UserControl上直接调用这个方法。

Page.Eval方法可以帮助我们更好的撰写数据绑定表达式,在ASP.NET 1.x时代,数据绑定表达式的一般形式是:

ExpandedBlockStart.gifContractedBlock.gif<% dot.gif# DataBinder.Eval( Container, " DataItem.Name "  )  %>


而在ASP.NET 2.0中,同样的代码,我们可以这样写:

ExpandedBlockStart.gifContractedBlock.gif<% dot.gif Eval ( " Name " %>


ASP.NET 2.0是怎么实现的呢?我们先从Eval方法来研究,通过反射.NET Framework 2.0类库的源代码,我们可以看到这个方法是这样实现的:

1 None.gifprotected   internal   object  Eval( string  expression)
2 ExpandedBlockStart.gifContractedBlock.gifdot.gif
{
3 InBlock.gif       this
.CheckPageExists();
4 InBlock.gif       return  DataBinder.Eval( this
.Page.GetDataItem(), expression);
5 ExpandedBlockEnd.gif}


第3行我们不必管,这是检查调用的时候有没有Page对象的,如果没有则会抛出一个异常。
关键是第4行:

None.gif       return  DataBinder.Eval( this .Page.GetDataItem(), expression);


Page.GetDataItem()也是2.0中新增的一个方法,用途是正是取代ASP.NET 1.x中的Container.DataItem。
看来不摸清楚GetDataItem()方法,我们也很难明白Eval的原理。GetDataItem的实现也很简单:

1 None.gifpublic   object  GetDataItem()
2 ExpandedBlockStart.gifContractedBlock.gifdot.gif
{
3 InBlock.gif       if  (( this ._dataBindingContext  ==   null ||  ( this ._dataBindingContext.Count  ==   0
))
4 ExpandedSubBlockStart.gifContractedSubBlock.gif       dot.gif
{
5 InBlock.gif             throw   new  InvalidOperationException(SR.GetString( " Page_MissingDataBindingContext "
));
6 ExpandedSubBlockEnd.gif      }

7 InBlock.gif      return   this ._dataBindingContext.Peek();
8 ExpandedBlockEnd.gif}


我们注意到了有一个内部对象_dataBindingContext,通过查源代码发现这是一个Stack类型的东西。所以他有Peek方法。而这一段代码很容易看懂,先判断这个Stack是否被实例化,然后,判断这个Stack里面是不是有任何元素,如果Stack没有被实例化或者没有元素则抛出一个异常。最后是将这个堆栈顶部的元素返回。

ASP.NET 2.0用了一个Stack来保存DataItem,我们很快就查到了为这个堆栈压元素和弹出元素的方法:Control.DataBind方法:


 1 None.gifprotected   virtual   void  DataBind( bool  raiseOnDataBinding)
 2 ExpandedBlockStart.gifContractedBlock.gifdot.gif
{
 3 InBlock.gif       bool  flag1  =   false ; // 这个标志的用处在上下文中很容易推出来,如果有DataItem压栈,则在后面出栈。

 4 InBlock.gif       if  ( this .IsBindingContainer) // 判断控件是不是数据绑定容器,实际上就是判断控件类是不是实现了INamingContainer
 5 ExpandedSubBlockStart.gifContractedSubBlock.gif       dot.gif{
 6 InBlock.gif             bool
 flag2;
 7 InBlock.gif             object  obj1  =  DataBinder.GetDataItem( this out  flag2); // 这个方法是判断控件是不是有DataItem属性,并把它取出来。

 8 InBlock.gif             if  (flag2  &&  ( this .Page  !=   null )) // 如果控件有DataItem
 9 ExpandedSubBlockStart.gifContractedSubBlock.gif             dot.gif{
10 InBlock.gif                   this .Page.PushDataBindingContext(obj1); // 把DataItem压栈,PushDataBindingContext就是调用_dataBindingContext的Push方法

11 InBlock.gif                  flag1  =   true ;
12 ExpandedSubBlockEnd.gif            }

13 ExpandedSubBlockEnd.gif      }

14 InBlock.gif       try
15 ExpandedSubBlockStart.gifContractedSubBlock.gif       dot.gif{
16 InBlock.gif             if  (raiseOnDataBinding) // 这里是判断是不是触发DataBinding事件的。

17 ExpandedSubBlockStart.gifContractedSubBlock.gif             dot.gif{
18 InBlock.gif                   this
.OnDataBinding(EventArgs.Empty);
19 ExpandedSubBlockEnd.gif            }

20 InBlock.gif             this .DataBindChildren(); // 对子控件进行数据绑定,如果这个控件有DataItem,则上面会将DataItem压入栈顶,这样,在子控件里面调用Eval或者GetDataItem方法,就会把刚刚压进去的DataItem给取出来。
21 ExpandedSubBlockEnd.gif      }

22 InBlock.gif       finally
23 ExpandedSubBlockStart.gifContractedSubBlock.gif       dot.gif{
24 InBlock.gif             if  (flag1) // 如果刚才有压栈,则现在弹出来。

25 ExpandedSubBlockStart.gifContractedSubBlock.gif             dot.gif{
26 InBlock.gif                   this .Page.PopDataBindingContext(); // PopDataBindingContext就是调用_dataBindingContext的Pop方法

27 ExpandedSubBlockEnd.gif            }

28 ExpandedSubBlockEnd.gif      }

29 ExpandedBlockEnd.gif}


至此,我们已经可以完全了解ASP.NET 2.0中GetDataItem和Eval方法运作的原理了,下一次我打算研究ASP.NET 2.0中的新的Bind语法。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值