Chinaunix首页 | 论坛 | 博客
  • 博客访问: 365411
  • 博文数量: 1051
  • 博客积分: 53280
  • 博客等级: 大将
  • 技术积分: 6670
  • 用 户 组: 普通用户
  • 注册时间: 2008-09-09 13:21
文章分类

全部博文(1051)

文章存档

2011年(1)

2008年(1050)

我的朋友

分类:

2008-09-09 15:45:03


  假定您在为一家新计算机公司起草业务计划,面对如何处理产品分发的问题。当然,您从未考虑过亲自将每一件新产品交付到每位客户的手中。相反,您会将此责任委托给像 FedEx 或 UPS 等配送服务机构。您必须事先做三个决定:该行动是什么(交付产品)、参数是什么(客户地址)以及返回值是什么(付款)。然后,当实际有一个包裹要交付时,您就可以将责任委托给某个特定的配送服务机构。
  
  在 Microsoft .NET 编程中,您会面对需要执行某种特定操作,但是无法预先知道将调用何种方法来执行该操作的情况。例如,在单击某个按钮时,可能需要调用某个方法,但是在设计该按钮时,却无法知道将调用哪个方法。您需要有一种方法来描述要调用何种方法。这正是委托派上用场的地方。在本文中,我将探索如何在 Visual Basic .NET 中使用委托。
  
  入门
  委托是一种引用类型,表示带有特定签名和返回类型的方法。可以在该委托中封装任何匹配的方法。另外,还可以将委托看作是对方法的引用。
  
  要了解其工作方式,我们来看一个委托解决的问题。正如您将在图 1 中看到的,我将创建一个 Pair 类的简单集合,其中有两个对象。Pair 类创建一个名为 thePair 的私有数组,其中包括两个成员:
  
  Public Class Pair
  
  Private thePair(2) As Object
  
  构造函数接受两个对象,然后按接收顺序将其添加到此内部数组中:
  
  Public Sub New(ByVal firstObject As Object, ByVal secondObject As Object)
  thePair(0) = firstObject
  thePair(1) = secondObject
  End Sub
  
  Pair 提供其它三种方法:Sort、ReverseSort 以及 ToString 的重写。Sort 方法将对内部数组中的两个对象进行排序。当然,您不希望让 Pair 类知道对这两个对象进行排序的测试,因为实际上您可能在 Pair 中任何类型的对象(Students、Dogs、Employees、Buttons,等等)。Pair 如何才能知道如何对所有这些不同类型的对象进行排序呢?
  
  解决方案是将责任(确定哪个对象最小)委托给对象自己。Pair 如何实现这一点?使用委托(参见图 1)。
  
 

  在 Pair 类中,我已经定义了一个委托以封装(引用)比较两个对象的方法,然后确定哪个较小(不管“较小”是如何定义的,确定的比较类都是合适的):
  
  Public Delegate Function WhichIsSmaller( _
   ByVal obj1 As Object, ByVal obj2 As Object) As Comparison
  
  这是一个相当复杂的定义。让我们来一段一段地查看这个定义:
  
  Public 关键字将该委托声明为 Pair 类的一个公共成员。
  
  Delegate 关键字表示您正在创建一个委托(而不是一个方法或属性)。
  
  Function 关键字表示该委托将用于封装一个函数(而不是一个子程序)。
  
  WhichIsSmaller 标识符是该委托的名称。
  
  括号内的值是该委托将封装的方法的签名。即该委托可能封装任何接受两个对象作为参数的函数。
  
  最后的关键字 As Comparison 是该委托可能封装的方法的返回类型。Comparison 是一个在 Pair 类中定义的枚举:
  
  Public Enum Comparison
  theFirst = 1
  theSecond = 2
  End Enum
  
  用该委托封装的方法必须返回 Comparison.theFirst 或 Comparison.theSecond。
  
  总之,刚刚展示的语句在名为 WhichIsSmaller 的 Pair 类中定义了一个公共委托,它封装接受两个对象作为参数的函数并返回 Comparison 枚举类型的一个实例。
  
  可以在该委托的实例中封装任何匹配的方法。例如,您的 Pair 集合可能包含两个 Student 对象,如下所示:
  
  Public Class Student
  
  Private name As String
  
  Public Sub New(ByVal name As String)
    Me.name = name
  End Sub
  
  ' other Student methods here
  
  End Class
  
  您的 Student 类必须创建一个与 WhichIsSmaller 委托相匹配的方法。例如,您可以创建一个 WhichStudentIsSmaller 方法(参见图 2 中的代码)。该方法匹配所要求的签名;它接受两个对象作为参数,并且返回一个 Comparison 值。
  
 

  因为我的 WhichStudentIsSmaller 方法需要将这些参数用作 Student 对象,而不是作为更一般的 Object 类型,我将把这些参数强制转换为 Student。这是类型的,因为我决不会用其他任何类型的参数来调用该方法。
  
  一旦我强制转换了这两个对象,我就可以对它们进行比较。在本例中,我将按字母顺序比较它们的名称值,并返回合适的枚举值:Pair.Comparison.theFirst 或 Pair.Comparison.theSecond。
  
  按字母顺序的比较是通过调用 String 类的共享方法 Compare 来完成的。如果按照字母表,第一个字符串排在第二个前面(s1.name 的字母顺序在 s2.name 之前),则 Compare 返回一个负整数值。如果按照字母表,第二个字符串排在第一个前面,则 Compare 返回一个正整数值,如果相同则返回零。
  
  其他类也能创建与 WhichIsSmaller 委托相匹配的方法。例如,我还可以创建 Dog 类:
  
  Public Class Dog
  
  Private weight As Integer
  
  Public Sub New(ByVal weight As Integer)
    Me.weight = weight
  End Sub
  
  ' other Dog methods here
  
  End Class
  
  该 Dog 类将实现一个方法,基于重量对两个 Dog 实例进行比较:
  
  Public Shared Function WhichDogIsSmaller( _
   ByVal o1 As Object, ByVal o2 As Object) As Pair.comparison
  Dim d1 As Dog = DirectCast(o1, Dog)
  Dim d2 As Dog = DirectCast(o2, Dog)
  If d1.weight > d2.weight Then
    Return Pair.Comparison.theSecond
  Else
    Return Pair.Comparison.theFirst
  End If
  End Function
  
  现在该 Pair 类就可以创建其 Sort 方法了。它将接受一个 WhichIsSmaller 委托作为参数:
  
  Public Sub Sort(ByVal theDelegatedFunc As WhichIsSmaller)
  If theDelegatedFunc(thePair(0), thePair(1)) = _
    Comparison.theSecond Then
    Dim temp As Object = thePair(0)
    thePair(0) = thePair(1)
    thePair(1) = temp
  End If
  End Sub
  
  Sort 方法通过委托来调用委托的方法,传递 Pair 数组的两个成员,返回一个枚举值。如果该值为 Comparison.theSecond,就知道第二个对象比第一个类型小。Sort 方法甚至无需知道这两个对象的类型就可以知道这一点!然后它可以将这两个对象颠倒过来。如果委托的方法返回 Comparison.theFirst,则无需。
  
  实例化委托
  要对此进行测试,可以创建两个 Student 对象,如下所示:
  
  Dim Jesse As New Student("Jesse")
  Dim Stacey As New Student("Stacey")
  
  然后将其添加到一个新的 Pair 对象中: Dim studentPair As New Pair(Jesse, Stacey)随后,可以实例化一个 WhichIsSmaller 委托,传递知道如何比较这两个 Student 对象的 Student 的匹配方法:
  
  Dim theStudentDelegate As New _
   Pair.WhichIsSmaller(AddressOf Student.WhichStudentIsSmaller)
  
  现在我可以将这个委托传递给 Sort 方法,以便对这个两个学生进行排序:
  
  studentPair.Sort(theStudentDelegate)
  
  类似地,我可以创建两个 Dog 对象,将其添加到 Pair,并基于比较两个 Dog 的 Dog 方法实例化一个委托,然后将该委托传递给 Dog 对的排序方法,如以下各行所示:
  
  ' make two dogs
  Dim Milo As New Dog(65)
  Dim Fred As New Dog(12)
  
  ' store the two dogs in a Pair
  Dim dogPair As New Pair(Milo, Fred)
  
  ' instantiate a delegate
  Dim theDogDelegate As New _
   Pair.WhichIsSmaller(AddressOf Dog.WhichDogIsSmaller)
  
  ' invoke Sort, pass in the delegate
  dogPair.Sort(theDogDelegate)
  
  Pair 类有一个接受相同的委托作为 Sort 的 ReverseSort。我同样可以将您刚刚创建的 Student 和 Dog 委托传递给 ReverseSort:
  
  studentPair.ReverseSort(theStudentDelegate)dogPair.ReverseSort(theDogDelegate)
  您可能已经猜到了,ReverseSort 的逻辑与 Sort 的逻辑相反。即,仅在比较方法返回的值指示第一个对象比第二个对象小的情况下,才 Pair 中的这两项。本例的完整源代码如图 1所示。
  
  实例方法与委托
  在图 1所示示例中,封装了 Dog 和 Student 类的共享方法。相反,可以同样容易地将封装的方法声明为实例方法(即,非共享方法):
  
  Public Comparison
  WhichStudentIsSmaller(o1 as Object, o2 as Object)
  
  尽管可以将该方法封装在委托中,但是必须通过实例而不是通过类来引用它,如下所示:
  
  Dim theStudentDelegate As _
   New Pai
【责编:admin】

--------------------next---------------------

阅读(140) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~