委托(Daelegate)delegate是C#中的一种类型,它实际上是一个能够持有对某个方法的引用的类。与其它的类不同,delegate类能够拥有一个签名(signature),并且它只能持有与它的签名相匹配的方法的引用。它所实现的功能与C/C++中的函数指针十分相似。它允许你传递一个类A的方法m给另一个类B的对象,使得类B的对象能够调用这个方法m。但与函数指针相比,delegate有许多函数指针不具备的优点。首先,函数指针只能指向静态函数,而delegate既可以引用静态函数,又可以引用非静态成员函数。在引用非静态成员函数时,delegate不但保存了对此函数入口指针的引用,而且还保存了调用此函数的类实例的引用。其次,与函数指针相比,delegate是面向对象、类型安全、可靠的受控(managed)对象。也就是说,runtime能够保证delegate指向一个有效的方法,你无须担心delegate会指向无效地址或者越界地址。
实现一个delegate是很简单的,通过以下3个步骤即可实现一个delegate:
step1.声明一个delegate类型,它应当与你想要传递的方法具有相同的参数和返回值类型。
step2.创建delegate对象,并将你想要传递的函数作为参数传入。
step3.在要实现异步调用的地方,通过上一步创建的对象来调用方法。
下面是一个简单的例子:
using System;
namespace MyApp
{
/**////
/// MyDelegateTest 的摘要说明。
///
public class MyDelegateTest
{
// 步骤1,声明delegate类型
public delegate void MyDelegate(string name);
// 这是我们欲传递的方法,它与MyDelegate具有相同的参数和返回值类型
public static void MyDelegateFunc(string name)
{
Console.WriteLine("静态方法: Hello, {0}", name);
}
public static void Main()
{
// 步骤2,创建delegate对象
MyDelegate md = new MyDelegate(MyDelegateTest.MyDelegateFunc);
// 步骤3,调用delegate
md("王铭");
//如何调用类成员方法
myclass mc=new myclass();
md=new MyDelegate(mc.mymethod);
md("Wang Ming");
//委托可以作为方法的参数
WL(md);
Console.ReadLine(); //为了显示控制台
}
public static void WL(MyDelegate mywl)
{
mywl("Ming Wang");
}
public MyDelegateTest()
{}
}
public class myclass
{
public void mymethod(string name)
{
Console.WriteLine("类成员方法: Hello,{0}",name);
}
}
}
下面是一个比较复杂的例子,使用委托技术,BubbleSorter.Sort方法可以排序任何数组,前提是数组成员类提供比较的方法,本例中Employee提供RhsIsGreater方法。
using System;
namespace Wrox.ProfCSharp.AdvancedCSharp
{
//步骤1 定义Delegated类型
delegate bool CompareOp(object lhs, object rhs);
class MainEntryPoint
{
static void Main()
{
Employee [] employees =
{
new Employee("Karli Watson", 20000),
new Employee("Bill Gates", 10000),
new Employee("Simon Robinson", 25000),
new Employee("Mortimer", (decimal)1000000.38),
new Employee("Arabel Jones", 23000),
new Employee("Avon from 'Blake's 7'", 50000)};
//步骤2 创建Delegate对象,传入要调用的函数作为参数
CompareOp employeeCompareOp = new CompareOp(Employee.RhsIsGreater);
BubbleSorter.Sort(employees, employeeCompareOp);
for (int i=0 ; i<employees.Length ; i++)
Console.WriteLine(employees[i].ToString());
Console.ReadLine();
}
}
class Employee // : object
{
private string name;
private decimal salary;
public Employee(string name, decimal salary)
{
this.name = name;
this.salary = salary;
}
public override string ToString()
{
return string.Format(name + ", {0:C}", salary);
}
public static bool RhsIsGreater(object lhs, object rhs)
{
Employee empLhs = (Employee) lhs;
Employee empRhs = (Employee) rhs;
return (empRhs.salary > empLhs.salary) ? true : false;
}
}
class BubbleSorter
{
static public void Sort(object [] sortArray, CompareOp gtMethod)
{
for (int i=0 ; i<sortArray.Length ; i++)
{
for (int j=i+1 ; j<sortArray.Length ; j++)
{
if (gtMethod(sortArray[j], sortArray[i]))//调用Delegate
{
object temp = sortArray[i];
sortArray[i] = sortArray[j];
sortArray[j] = temp;
}
}
}
}
}
}