デリゲートの使用 (C# プログラミング ガイド)

デリゲートは、C および C++ の関数ポインターのようなメソッドを安全にカプセル化する型です。 ただし、C 関数ポインターとは異なり、デリゲートはオブジェクト指向で、タイプ セーフで、安全です。 デリゲートの型は、デリゲートの名前によって定義されます。 次の例では、引数として文字列を受け取り、void を返すメソッドをカプセル化できる Callback という名前のデリゲートを宣言しています。

public delegate void Callback(string message);

デリゲート オブジェクトは、通常、デリゲートによってラップされるメソッドの名前を指定するかラムダ式を使用することで構成されます。 この方法でインスタンス化されたデリゲートは呼び出すことができます。 デリゲートを呼び出すと、デリゲート インスタンスにアタッチされているメソッドが呼び出されます。 呼び出し元によってデリゲートに渡されるパラメーターはメソッドに渡され、戻り値がある場合は、デリゲートによってメソッドから呼び出し元に返されます。 次に例を示します。

// Create a method for a delegate.
public static void DelegateMethod(string message)
{
    Console.WriteLine(message);
}
// Instantiate the delegate.
Callback handler = DelegateMethod;

// Call the delegate.
handler("Hello World");

デリゲート型は、.NET の Delegate クラスから派生しています。 デリゲート型は sealed (派生できません) であり、Delegate からカスタム クラスを派生することはできません。 インスタンス化されたデリゲートはオブジェクトであるため、引数として渡したり、プロパティに割り当てたりできます。 これにより、メソッドは、パラメーターとしてデリゲートを受け入れ、後でデリゲートを呼び出すことができます。 これは非同期のコールバックと呼ばれ、長いプロセスの完了時に呼び出し元に通知する一般的な方法です。 デリゲートをこの方法で使用する場合、デリゲートを使用するコードは、使用されるメソッドの実装について認識している必要はありません。 機能は、カプセル化インターフェイスが提供する機能に似ています。

コールバックの別の一般的な使用方法は、カスタム比較メソッドの定義およびそのデリゲートの並べ替えメソッドへの引き渡しです。 これにより、呼び出し元のコードを並べ替えアルゴリズムの一部にすることができます。 次の例のメソッドは Del 型をパラメーターとして使用しています。

public static void MethodWithCallback(int param1, int param2, Callback callback)
{
    callback("The number is: " + (param1 + param2).ToString());
}

上記で作成したデリゲートをメソッドに渡すことができます。

MethodWithCallback(1, 2, handler);

次の出力がコンソールに表示されます。

The number is: 3

抽象化としてデリゲートを使用する場合、MethodWithCallback がコンソールを直接呼び出す必要はありません。コンソールに留意して設計する必要はありません。 MethodWithCallback は文字列を準備し、文字列を別のメソッドに渡すだけです。 これは、デリゲート メソッドが任意の数のパラメーターを使用できるため、特に強力です。

インスタンス メソッドをラップするようにデリゲートが構築された場合、デリゲートはインスタンスとメソッドの両方を参照します。 デリゲートはラップするメソッド以外のインスタンス型を認識しないため、デリゲート シグネチャと一致するオブジェクトにメソッドがある限り、どの型のオブジェクトでも参照できます。 静的メソッドをラップするようにデリゲートが構築された場合、デリゲートはそのメソッドのみを参照します。 次に宣言の例を示します。

public class MethodClass
{
    public void Method1(string message) { }
    public void Method2(string message) { }
}

以前に示した静的な DelegateMethod と共に、Del インスタンスによってラップできるメソッドが 3 つあります。

デリゲートは、呼び出されたときに複数のメソッドを呼び出すことができます。 これはマルチキャスティングと呼ばれます。 デリゲートのメソッドの一覧 (呼び出しリスト) に追加のメソッドを追加するには、加算演算子または加算代入演算子 ('+' または '+=') を使用して 2 つのデリゲートを追加する必要があります。 次に例を示します。

var obj = new MethodClass();
Callback d1 = obj.Method1;
Callback d2 = obj.Method2;
Callback d3 = DelegateMethod;

//Both types of assignment are valid.
Callback allMethodsDelegate = d1 + d2;
allMethodsDelegate += d3;

この時点で、allMethodsDelegate には、呼び出しリスト内の 3 つのメソッド (Method1Method2DelegateMethod) が含まれています。 元の 3 つのデリゲートである d1d2、および d3 は変更されません。 allMethodsDelegate が呼び出されると、すべての 3 つのメソッドが順に呼び出されます。 デリゲートで参照パラメーターを使用する場合、参照は、3 つのメソッドに順番に渡され、1 つのメソッドによって行われた変更は、次のメソッドに示されます。 いずれかのメソッドがメソッド内でキャッチされない例外をスローした場合、デリゲートの呼び出し元に例外が渡され、呼び出しリスト内の後続のメソッドは呼び出されません。 デリゲートに戻り値や out パラメーターがある場合、デリゲートは戻り値と最後に呼び出されたメソッドのパラメーターを返します。 呼び出しリストからメソッドを削除するには、減算演算子または減算代入演算子 (- または -=) を使います。 次に例を示します。

//remove Method1
allMethodsDelegate -= d1;

// copy AllMethodsDelegate while removing d2
Callback oneMethodDelegate = allMethodsDelegate - d2;

デリゲート型が System.Delegate から派生しているため、そのクラスで定義されるメソッドとプロパティはデリゲートで呼び出すことができます。 たとえば、デリゲートの呼び出しリスト内のメソッドの数を検索するには、次のように記述します。

int invocationCount = d1.GetInvocationList().GetLength(0);

呼び出しリストに複数のメソッドがあるデリゲートは、MulticastDelegate のサブクラスである System.Delegate から派生します。 上記のコードでは、両方のクラスが GetInvocationList をサポートしているため、いずれの場合も機能します。

マルチキャスト デリゲートは、イベント処理で広く使用されます。 イベント ソース オブジェクトはイベントを受け取るように登録されている受信者オブジェクトにイベント通知を送信します。 イベントを登録するには、受信者は、イベントを処理するようにデザインされたメソッドを作成し、そのメソッドのデリゲートを作成してイベント ソースにデリゲートを渡します。 イベントが発生すると、ソースがデリゲートを呼び出します。 デリゲートは、受信者のイベント処理メソッドを呼び出し、イベント データを配信します。 特定のイベントのデリゲート型は、イベント ソースによって定義されます。 詳細については、「イベント (C# プログラミング ガイド)」を参照してください。

コンパイル時に割り当てられた 2 つの異なる型のデリゲートを比較すると、コンパイル エラーが発生します。 デリゲート インスタンスが静的な System.Delegate 型の場合は、比較できますが、実行時に false が返されます。 次に例を示します。

delegate void Callback1();
delegate void Callback2();

static void method(Callback1 d, Callback2 e, System.Delegate f)
{
    // Compile-time error.
    //Console.WriteLine(d == e);

    // OK at compile-time. False if the run-time type of f
    // is not the same as that of d.
    Console.WriteLine(d == f);
}

関連項目