C#委托 Delegate
C#中的委托类似于C或C++中的函数指针,它是存有对某个方法的引用的引用类型变量,引用可在运行时被改变。
委托都派生自System.Delegate类
- 声明委托
委托声明决定了可由该委托引用的方法。
声明委托的语法:
delegate <delegate_name>();
比如有一个委托如下:
public delegate int MyDelegate(string s);
上面的这个委托可被用于引用任何一个带有一个string参数的方法,并且返回一个int类型变量。
- 实例化委托
当声明了委托类型,委托对象必须要使用new关键字来创建,并且与一个特定的方法有关。当创建委托时,传递到new语句的参数就像方法调用一样书写,但是不带参数。
比如:
public delegate void Todo(string s)
Todo td1 = new Todo(say);
Todo td2 = new Todo(eat);
使用Visual Studio新建C#控制台应用程序chapter22_005
//声明一个委托
delegate int NumberChanger(int n);
class Program
{
static int num = 10;
//相加
public static int AddNum(int p)
{
num += p;
return num;
}
//相乘
public static int MulNum(int q)
{
num *= q;
return num;
}
public static int GetNum()
{
return num;
}
static void Main(string[] args)
{
//创建委托实例
NumberChanger nc_add = new NumberChanger(AddNum);
NumberChanger nc_mul = new NumberChanger(MulNum);
//使用委托对象来调用方法
nc_add(10);
Console.WriteLine("使用nc_add委托对象调用nc_add(10)后num的值:{0}", GetNum());
nc_mul(2);
Console.WriteLine("使用nc_mul委托对象调用nc_mul(2)后num的值:{0}", GetNum());
Console.ReadKey();
}
}
编译运行后的结果如下:
- 委托的多播
委托对象可使用"+“运算符进行合并,一个合并委托调用它所合并的两个委托。只有相同类型的委托可以合并。”-"运算符可用于从合并的委托中移除组件委托。
使用委托的这个特点,可以创建一个委托调用时要调用的方法的调用列表。这个称为委托的多播,也叫组播。
在上面Main方法中加入如下代码进行测试:
//创建委托实例
NumberChanger nc;
//委托多播
nc = nc_add;
nc += nc_mul;
//调用多播
nc(5);
Console.WriteLine("进行多播调用后num的值:{0}",GetNum());
最后运行结果如下:
- 委托用途
下面的实例来把委托做为某个方法的参数,通过传入不同的委托实例来做调用。
使用Visual Studio 新建C#控制台应用程序chapter22_006
添加如下代码:
class Program
{
static FileStream fs;
static StreamWriter sw;
//委托声明
public delegate void printString(string s);
//控制台打印方法
public static void WriteToConsole(string str)
{
Console.WriteLine("打印内容:{0}",str);
}
//输出到文件
public static void WriteToFile(string s)
{
fs = new FileStream("..\\..\\message.txt", FileMode.Append, FileAccess.Write);
sw = new StreamWriter(fs);
sw.WriteLine(s);
sw.Flush();
sw.Close();
fs.Close();
}
//把委托为为参数,并使用参数中的委托进行调用
public static void sendString(printString ps)
{
ps("Hello,C#");
}
static void Main(string[] args)
{
printString ps1 = new printString(WriteToConsole);
printString ps2 = new printString(WriteToFile);
sendString(ps1);
sendString(ps2);
Console.ReadKey();
}
}
编译运行后的结果如下: