合并委托
本示例演示如何创建多播委托。 委托对象的一个有用属性是:可以使用 + 运算符将多个对象分配给一个委托实例。多播委托包含已分配委托的列表。在调用多播委托时,它会按顺序调用列表中的委托。只能合并相同类型的委托。
- 运算符可用于从多播委托中移除组件委托。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
|
using System;
// Define a custom delegate that has a string parameter and returns void.
delegate void CustomDel( string s);
class TestClass
{
// Define two methods that have the same signature as CustomDel.
static void Hello( string s)
{
System.Console.WriteLine( " Hello, {0}!" , s);
}
static void Goodbye( string s)
{
System.Console.WriteLine( " Goodbye, {0}!" , s);
}
static void Main()
{
// Declare instances of the custom delegate.
CustomDel hiDel, byeDel, multiDel, multiMinusHiDel;
// In this example, you can omit the custom delegate if you
// want to and use Action<string> instead.
//Action<string> hiDel, byeDel, multiDel, multiMinusHiDel;
// Create the delegate object hiDel that references the
// method Hello.
hiDel = Hello;
// Create the delegate object byeDel that references the
// method Goodbye.
byeDel = Goodbye;
// The two delegates, hiDel and byeDel, are combined to
// form multiDel.
multiDel = hiDel + byeDel;
// Remove hiDel from the multicast delegate, leaving byeDel,
// which calls only the method Goodbye.
multiMinusHiDel = multiDel - hiDel;
Console.WriteLine( "Invoking delegate hiDel:" );
hiDel( "A" );
Console.WriteLine( "Invoking delegate byeDel:" );
byeDel( "B" );
Console.WriteLine( "Invoking delegate multiDel:" );
multiDel( "C" );
Console.WriteLine( "Invoking delegate multiMinusHiDel:" );
multiMinusHiDel( "D" );
}
}
|
输出:
1
2
3
4
5
6
7
8
9
|
Invoking delegate hiDel:
Hello, A!
Invoking delegate byeDel:
Goodbye, B!
Invoking delegate multiDel:
Hello, C!
Goodbye, C!
Invoking delegate multiMinusHiDel:
Goodbye, D!
|
声明、实例化和使用委托
在 C# 1.0 及更高版本中,可以按以下示例所示声明委托。
1
2
3
4
5
6
7
8
9
10
11
12
|
// Declare a delegate.
delegate void Del( string str);
// Declare a method with the same signature as the delegate.
static void Notify( string name)
{
Console.WriteLine( "Notification received for: {0}" , name);
}
// Create an instance of the delegate.
Del del1 = new Del(Notify);
|
C# 2.0 提供了更简单的方法来编写上面的声明,如以下示例所示。
1
2
|
// C# 2.0 provides a simpler way to declare an instance of Del.
Del del2 = Notify;
|
在 C# 2.0 及更高版本中,还可以使用匿名方法来声明和初始化委托,如以下示例所示。
1
2
3
|
// Instantiate Del by using an anonymous method.
Del del3 = delegate ( string name)
{ Console.WriteLine( "Notification received for: {0}" , name); };
|
在 C# 3.0 及更高版本中,还可以使用 Lambda 表达式来声明和实例化委托,如以下示例所示。
1
2
|
// Instantiate Del by using a lambda expression.
Del del4 = name => { Console.WriteLine( "Notification received for: {0}" , name); };
|
下面的示例阐释声明、实例化和使用委托。 BookDB 类封装一个书店数据库,它维护一个书籍数据库。它公开 ProcessPaperbackBooks 方法,该方法在数据库中查找所有平装书,并对每本平装书调用一个委托。使用的 delegate 类型名为 ProcessBookDelegate。 Test 类使用该类打印平装书的书名和平均价格。
委托的使用促进了书店数据库和客户代码之间功能的良好分隔。客户代码不知道书籍的存储方式和书店代码查找平装书的方式。书店代码也不知道找到平装书后将对平装书执行什么处理。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
|
// A set of classes for handling a bookstore:
namespace Bookstore
{
using System.Collections;
// Describes a book in the book list:
public struct Book
{
public string Title; // Title of the book.
public string Author; // Author of the book.
public decimal Price; // Price of the book.
public bool Paperback; // Is it paperback?
public Book( string title, string author, decimal price, bool paperBack)
{
Title = title;
Author = author;
Price = price;
Paperback = paperBack;
}
}
// Declare a delegate type for processing a book:
public delegate void ProcessBookDelegate(Book book);
// Maintains a book database.
public class BookDB
{
// List of all books in the database:
ArrayList list = new ArrayList();
// Add a book to the database:
public void AddBook( string title, string author, decimal price, bool paperBack)
{
list.Add( new Book(title, author, price, paperBack));
}
// Call a passed-in delegate on each paperback book to process it:
public void ProcessPaperbackBooks(ProcessBookDelegate processBook)
{
foreach (Book b in list)
{
if (b.Paperback)
// Calling the delegate:
processBook(b);
}
}
}
}
// Using the Bookstore classes:
namespace BookTestClient
{
using Bookstore;
// Class to total and average prices of books:
class PriceTotaller
{
int countBooks = 0;
decimal priceBooks = 0.0m;
internal void AddBookToTotal(Book book)
{
countBooks += 1;
priceBooks += book.Price;
}
internal decimal AveragePrice()
{
return priceBooks / countBooks;
}
}
// Class to test the book database:
class TestBookDB
{
// Print the title of the book.
static void PrintTitle(Book b)
{
System.Console.WriteLine( " {0}" , b.Title);
}
// Execution starts here.
static void Main()
{
BookDB bookDB = new BookDB();
// Initialize the database with some books:
AddBooks(bookDB);
// Print all the titles of paperbacks:
System.Console.WriteLine( "Paperback Book Titles:" );
// Create a new delegate object associated with the static
// method Test.PrintTitle:
bookDB.ProcessPaperbackBooks(PrintTitle);
// Get the average price of a paperback by using
// a PriceTotaller object:
PriceTotaller totaller = new PriceTotaller();
// Create a new delegate object associated with the nonstatic
// method AddBookToTotal on the object totaller:
bookDB.ProcessPaperbackBooks(totaller.AddBookToTotal);
System.Console.WriteLine( "Average Paperback Book Price: ${0:#.##}" ,
totaller.AveragePrice());
}
// Initialize the book database with some test books:
static void AddBooks(BookDB bookDB)
{
bookDB.AddBook( "The C Programming Language" , "Brian W. Kernighan and Dennis M. Ritchie" , 19.95m, true );
bookDB.AddBook( "The Unicode Standard 2.0" , "The Unicode Consortium" , 39.95m, true );
bookDB.AddBook( "The MS-DOS Encyclopedia" , "Ray Duncan" , 129.95m, false );
bookDB.AddBook( "Dogbert's Clues for the Clueless" , "Scott Adams" , 12.00m, true );
}
}
}
|
输出:
1
2
3
4
5
|
Paperback Book Titles:
The C Programming Language
The Unicode Standard 2.0
Dogbert's Clues for the Clueless
Average Paperback Book Price: $23.97
|
可靠编程
声明委托。
下面的语句声明一个新的委托类型。
1
|
public delegate void ProcessBookDelegate(Book book);
|
每个委托类型都描述参数的数目和类型,以及它可以封装的方法的返回值类型。每当需要一组新的参数类型或新的返回值类型时,都必须声明一个新的委托类型。
实例化委托。
声明了委托类型后,必须创建委托对象并使之与特定方法关联。在上一个示例中,您通过按下面示例中的方式将 PrintTitle 方法传递到 ProcessPaperbackBooks 方法来实现这一点:
1
|
bookDB.ProcessPaperbackBooks(PrintTitle);
|
这将创建与静态方法 Test.PrintTitle 关联的新委托对象。类似地,对象 totaller 的非静态方法 AddBookToTotal 是按下面示例中的方式传递的:
1
|
bookDB.ProcessPaperbackBooks(totaller.AddBookToTotal);
|
在两个示例中,都向 ProcessPaperbackBooks 方法传递了一个新的委托对象。
委托创建后,它的关联方法就不能更改;委托对象是不可变的。
调用委托。
创建委托对象后,通常将委托对象传递给将调用该委托的其他代码。通过委托对象的名称(后面跟着要传递给委托的参数,括在括号内)调用委托对象。下面是委托调用的示例:
1
|
processBook(b);
|
与本例一样,可以通过使用 BeginInvoke 和 EndInvoke 方法同步或异步调用委托。