web-dev-qa-db-ja.com

C#での適切なカリー化

(パラメータのない)関数を取り、それを何らかの方法で処理するメソッドDoSomethingが与えられます。以下のスニペットよりも、パラメーターを使用して関数の「オーバーロード」を作成するためのより良い方法はありますか?

public static TResult DoSomething<TResult>(Func<TResult> func)
{
    //call func() and do something else
}

public static TResult DoSomething<T0, TResult>(
    Func<T0, TResult> func,
    T0 arg0)
{
    return DoSomething(() => func(arg0));
}

public static TResult DoSomething<T0, T1, TResult>(
    Func<T0, T1, TResult> func,
    T0 arg0, T1 arg1)
{
    return DoSomething(arg => func(arg, arg1), arg0);
}

public static TResult DoSomething<T0, T1, T2, TResult>(
    Func<T0, T1, T2, TResult> func,
    T0 arg0, T1 arg1, T2 arg2)
{
    return DoSomething(arg => func(arg, arg1, arg2), arg0);
}
33
Rauhotz

編集:コメントに記載されているように、これはカリー化ではなく部分適用です。私は 違いの理解に関するブログ投稿 を書きました。これは人々が面白いと思うかもしれません。

まあ、それは特に違いはありませんが、カリー化の部分を「DoSomethingの呼び出し」の部分から分離します。

public static Func<TResult> Apply<TResult, TArg> (Func<TArg, TResult> func, TArg arg)
{
    return () => func(arg);
}

public static Func<TResult> Apply<TResult, TArg1, TArg2> (Func<TArg1, TArg2, TResult> func,
                                                          TArg1 arg1, TArg2 arg2)
{
    return () => func(arg1, arg2);
}

// etc

次に:

DoSomething(Apply(foo, 1));

そうすれば、新しく返されたデリゲートをすぐに呼び出したくない場合など、他の状況でカリー化コードを再利用できます。 (たとえば、後でもっとカレーしたいと思うかもしれません。)

31
Jon Skeet