[例程]Delegate声明的定义使用 作者: Builder.comThursday, August 12 2004 3:22 PM
|
如何定义一个delegate并且在何处使用?Tony Patton在本文中深入讨论了delegate。
如何定义一个delegate并且在何处使用?Tony Patton在本文中深入讨论了delegate。
在最近有关.NET项目一次技术采访中,我被问到了关于定义术语的问题。由于项目是采用C#的.NET平台,所以问题就集中在C#上面。采访的过程非常有趣,但有一个特别的问题却是使我兴致大减,这一问题是:我是否必须定义delegate。虽然我已经广泛地使用它,但我还是很难从定义上回答。尽管我已经尽力回答,我还是感到事实尴尬。事实上,一个MCSD.NET应该能够毫无疑问地回答这样的问题。
delegate是什么?
根据微软的说法,一个delegate声明指定了一个相关的类型,你可以使用这一类型来封装一个含有特定标记的方法。一个delegate实例可以封装一个静态或一个实例方法。粗略的看来,Delegates就好象是C++中的函数指针;然而,delegates属于安全和保护类型。
我的采访是通过电话进行,所以我可以很容易和轻松地阐述这些定义。透明和详尽的解释是必须的。然而,作为贴切的delegates定义是,即使在没有对目标对象(target object)有任何了解的情况下,也可以随时使用某种方法,这样的方法即为delegates。很多开发人员通常将它们说成回调函数(callback functions)。现在让我们看一个范例,以更好理解它的应用。
应用中的Delegates
我们的范例程序开始于两个简单的C#类,这两个类包含一个方法以通过控制台来显示信息:
public class DelegateClass1 {
public void exampleMethod1(String passedString) {
Console.WriteLine(passedString); }
}
public class DelegateClass2 {
public void exampleMethod2(String passedString) {
Console.WriteLine(passedString); }
} |
注意这两个类执行之间类似的方法。这些方法具有不同的名称,但它们的标记和实体都相同。你可以使用一个delegate来将这两个方法结合在一起,即通过定义一个从这两个方法的共同特性中抽象出新的数据类型。也就是它们都将一个简单的字符作为一个变量并返回void。C#中的delegate关键词使得一下语句成为可能:
public delegate void example(String passedString);
例如,使用这一新的数据类型声明,我们可以通过一个通用的抽象来调用这两个方法。下面的C#控制台程序利用了我们的delegate范例来调用这两个类的方法:
public class TestExampleDelegate {
public delegate void example(String passedString);
public static void Main(string[] args) {
example[] items = new example[2];
items[0] = new example(new DelegateClass1().exampleMethod1);
items[1] = new example(new DelegateClass2().exampleMethod2);
for(int i = 0; i < items.Length; i++) {
items[i]("Builder.com Rules!");
} } }
以上代码建立了一个数组的范例对象,并使用这些类中的方法实例化这些对象。由于忽略了括号,范例对象被一个调用方法处理。代码的执行结果显示了文本中(两次)。
(注意:Delegates不属于C#的细节;他们是.NET Framework的标准部分,所以也你可以使用其它的语言,比如VB.NET或者J#。)
Delegate无处不在
现在,也许你理解了它的原理,但却没有察觉它在你的程序代码中是如何应用的。尽管你没有建立自己的delegates,但如果你经常使用事件,你就一定会使用到他们。一个极好的例子就是ASP.NET中使用的事件。在后台使用的Delegates围绕着事件工作。事件使用一个一致标记,从而使delegate可用,你也可以在一个线程结构中找到delegates。
接口
你必须注意的另一个问题是一个delegate和一个接口之间的相似性,因为它们都将规范和执行分离开来。它们允许多个开发人员建立与规范相符合的执行。除此之外,方法标记必须被指定并且相互保持一致。
所以,在什么时候使用接口?以及什么时候使用delegates?微软提供了以下的指导思想。
在以下情况下使用Delegates很有用:
调用一个单一方法;
一个类要进行方法规范(method specification)的多种执行;
使用一个静态方法来执行规范;
想获得类似事件设计的模式;
调用者没有必要知道或获得方法定义的对象;
执行的提供者想将规范的执行“分发(hand out)”成一些可供选择的部分;
代码需要进行简单的组成部分。
接口在以下情况会很有用:
规范指定一套即将被调用的方法;
特别的,一个类只执行规范一次;
接口的调用者想通过接口类型以获得其它接口或类。
知识就是力量
这些关于开发的问题被谈论得如此严肃似乎有些可笑,但给出明确的定义或解释常常会比较困难。对于delegate的定义就是一个很好的例子。对于delegate特点的概述希望能够帮助你注意应当在何时何地使用它。同时,你也能够应付关于此类问题的采访。
本文作者:Tony Patton是作为一名应用程序开发人员开始其技术职业生涯的。他所通过的关于Java、VB、Lotus以及XML的认证大大的提高了他的技术水平。
[C#]
using System;
public class SamplesDelegate {
// Declares a delegate for a method that takes in an int and returns a String.
public delegate String myMethodDelegate( int myInt );
// Defines some methods to which the delegate can point.
public class mySampleClass {
// Defines an instance method.
public String myStringMethod ( int myInt ) {
if ( myInt > 0 )
return( "positive" );
if ( myInt < 0 )
return( "negative" );
return ( "zero" );
}
// Defines a static method.
public static String mySignMethod ( int myInt ) {
if ( myInt > 0 )
return( "+" );
if ( myInt < 0 )
return( "-" );
return ( "" );
}
}
public static void Main() {
// Creates one delegate for each method.
mySampleClass mySC = new mySampleClass();
myMethodDelegate myD1 = new myMethodDelegate( mySC.myStringMethod );
myMethodDelegate myD2 = new myMethodDelegate( mySampleClass.mySignMethod );
// Invokes the delegates.
Console.WriteLine( "{0} is {1}; use the sign \"{2}\".", 5, myD1( 5 ), myD2( 5 ) );
Console.WriteLine( "{0} is {1}; use the sign \"{2}\".", -3, myD1( -3 ), myD2( -3 ) );
Console.WriteLine( "{0} is {1}; use the sign \"{2}\".", 0, myD1( 0 ), myD2( 0 ) );
}
}