附源代码+送分。通过编写自定义控件实例再论Delegate, Event Handling, anonymous method。
用一个开发实例再谈谈Delegate和Event Handling. 代码实现一个可排序的DropDownList自定义控件,分布讲解。
自定义控件的名称: SortableDropDownList, 继承DropDownList
不再过多解释什么是Delegate 和 Event Handling,要说的是,实战中,为什么要用它们,如何用它们提高开发效率,而不是为了用而用。
基于ASP.NET2.0的开发中,经常有二个场景使用Delegate:
1. Delegate + Event Handling这个场景是经典Delegate使用场景,在这个场景中Delegate 和 Event Handling是联系在一起的。使用它的目的是将类更好的封装,定义这个类的事件,通过Delegate建立一个通道,当事件产生时可以传送给调用这个类的另外一个类。下面举例:
设计SortableDropDownList时,假设需要一个事件,这个事件是排序前激发,不返回任何EventArg.
首先定义事件: Public public event EventHandler SortingBegin; (这里没有定义delegate,因为EventHandler本身就是一个无返回事件数据的delegate,.Net已帮你定义好了。)
接着定义这个事件的方法:
C# code
public void OnSortingBegin()
{
if (SortingBegin != null)
{
SortingBegin(this, new EventArgs());
}
}
然后考虑什么时候调OnSortingBegin()这个方法以激发事件?自定义控件继承自dropdownlist,那么dropdownlist的PreRender事件是asp.net生命周期的最后一个事件,所以在这个事件激发的时候,dropdownlist所需做的都已经做完了,数据也已经绑定。所以就override这个事件。
C# code
protected override void OnPreRender(EventArgs e)
{
base.OnPreRender(e); //做这个dropdown应该做的事,先。
OnSortingBegin(); //排序前调用事件方法
SortItems(); //排序,忽略这个,先。
}
现在,在某个使用这个控件的aspx页面,这个控件的属性里你就可以加上一个<..... SortingBegin = "OnSrotBegin"...>, 就像一个button,属性里你可以加 <...OnClick="...."> 一样。然后假设需要定义另外一个事件,在排序完成时激发,返回排序前的itemList和排序后的itemList.
那么我首先定义这个事件返回数据类:public class SortingCompleteEventArgs : EventArgs (任何事件返回数据类都必须继承EventArgs类)
完整代码:
C# code
public class SortingCompleteEventArgs : EventArgs
{
private ListItem[] originalItems;
private ListItem[] sortedItems;
public ListItem[] OriginalItems
{
set { originalItems = value; }
get { return originalItems; }
}
public ListItem[] SortedItems
{
set { sortedItems = value; }
get { return sortedItems; }
}
public SortingCompleteEventArgs(int length)
{
originalItems = new ListItem[length];
sortedItems = new ListItem[length];
}
}
然后在自定义控件的类里加入如下代码,并做第一个事件代码的修改补充:
C# code
public delegate void SortingCompleteHandler(object sender, SortingCompleteEventArgs arg);
public event SortingCompleteHandler SortingComplete;
public void OnSortingComplete(SortingCompleteEventArgs arg)
{
if (SortingComplete != null)
{
SortingComplete(this, arg);
}
}
protected override void OnPreRender(EventArgs e)
{
base.OnPreRender(e);
SortingCompleteEventArgs arg = new SortingCompleteEventArgs(this.Items.Count);
Items.CopyTo(arg.OriginalItems, 0);
OnSortingBegin(); //调用事件1的方法。
SortItems(); //排序方法,忽略,先。
Items.CopyTo(arg.SortedItems, 0);
OnSortingComplete(arg); //调用事件2的方法。
}
下面来说使用Delegate的第二场景 -- Delegate + 匿名方法匿名方法是.Net 2.0后特有,作用是一小段代码,经常用,却不值得为它定义一个方法,那么就用匿名方法,
举个例--排序。
一个实体类Customer, 属性有Id(int),Name(string). 现在有List<Customer> customers已经取值完成。
现在来玩delegate + 匿名方法,
按照Name排序(升序):
customers.Sort(delegate(customer c1, customer c2) { return c1.Name.CompareTo(c2.Name) ;} );
降序:
customers.Sort(delegate(customer c1, customer c2) { return c2.Name.CompareTo(c1.Name) ;} );
按照Id排序:
customers.Sort(delegate(customer c1, customer c2) { return c1.Id > c2.Id ;} );