有名な C#
言語の第一人者 であり、 C#
In Depthの 著者であり 、Googleの従業員であり、 stackoverflow.comの評判の 人物 #1 であり、最後にヒーローのJon Skeet Facts による記事の翻訳 。 この記事では、関数プログラミングの世界から来た概念である、関数のカリー化と部分適用とは何かを明確に説明しています。 さらに、彼はその違いが何であるかを詳細に説明します。 この記事を読む前に私自身がそれらを混同していたことを認めているので、翻訳をすることは私にとって有用であるように思えました。
これは少し奇妙な投稿です。読む前に、おそらく次のグループのいずれかを参照する必要があります。
- 関数型プログラミングに興味がなく、高階関数がわかりにくい場合:この記事は完全にスキップできます。
- 関数型プログラミングについてすべてを知っており、カリー化と部分関数アプリケーションの違いをよく理解している人:この投稿を注意深く読んで、不正確な点があればコメントで登録を解除してください。
- 関数型プログラミングにある程度精通しており、さらに学習することに興味がある人:この投稿に懐疑的で、コメントを注意深く読んでください。 詳細については、経験豊富な開発者による他の記事を参照してください。
一般に、
関数の カリー化と
部分的な適用という用語を混同する人もいることを知っています。これは、行うべきではないときに同じ意味で使用されます。 これは、ある程度理解しているトピック(モナドなど)の1つであり、自分の知識を検証する最良の方法はそれについて書くことだと判断しました。 これにより、トピックが他の開発者にとってよりアクセスしやすくなる場合は、さらに良いでしょう。
この投稿にはHaskellは含まれていません。
私が見たこの主題に関するほとんどすべての説明で、例は「正しい」関数言語、通常はHaskellで与えられました。 Haskellに反対するものは何もありません。通常、私がよく知っている言語の例を理解する方が簡単です。 さらに、このような言語で例を
書く方が
はるかに簡単なので、この投稿のすべての例はC#で
記述されます。 実際、すべての例は
1つのファイルで利用でき
ますが 、その中のいくつかの変数は名前が変更されています。 コンパイルして実行するだけです。
C#は実際には関数型言語ではありません-デリゲートは高階関数の完全な代替物ではないことを十分理解しています。 ただし、説明されている原則を実証するには十分です。
カリー化と部分的なアプリケーションは、少数の引数を持つ関数(メソッド)を使用して実証できますが、明確にするために3つの引数を使用することにしました。 カリー化と部分的なアプリケーションを実行するための私のメソッドは一般化されます(したがって、すべてのタイプのパラメーターと戻り値は任意です)が、デモのために単純な関数を使用します。
static string SampleFunction(int a, int b, int c) { return string.Format("a={0}; b={1}; c={2}", a, b, c); }
これまでのところ、すべてがシンプルです。 この方法に
はcなものは何もありませんが、驚くべきことを探してはいけません。
それは何ですか?
カリー化と部分適用の両方は、ある種の機能を別の種類に変換する方法です。 関数の近似としてデリゲートを使用するため、SampleFunctionメソッドを値として使用するには、次のように記述できます。
Func<int, int, int, string> function = SampleFunction;
この行は、次の2つの理由で役立ちます。
- 変数に値を割り当てると、それが実際に値であるというアイデアが生まれます。 デリゲートインスタンスは、他の型のインスタンスと同様にオブジェクトであり、関数変数の値は単なる参照です。
- メソッドのグループの変換(デリゲートを作成する方法としてメソッド名を使用)は、ジェネリックメソッドが呼び出されたときの型推論ではうまく機能しません。
これで、デリゲートを3つの引数で呼び出すことができます。
string result = function(1, 2, 3);
または同じこと:
string result = function.Invoke(1, 2, 3);
(C#コンパイラは、最初の短い形式を2番目の形式に変換します。生成されるILは同じになります。)
さて、3つの引数すべてを一度に使用できる場合はどうでしょうか。 特定の(やや不自然ですが)例については、3つのパラメーター(ソース、重大度、メッセージ)を持つロギング関数があり、同じクラス(これをBusinessLogicと呼びます)内で、常に同じ値をパラメーターに使用したいとします「ソース」。 クラスのどこからでも簡単にログを記録できるようにして、重大度とメッセージのみを示します。 いくつかのオプションがあります:
- ロギング関数(またはロガーオブジェクト)とそのコンストラクターのソースパラメーターの値を受け入れ、それらをフィールドに保存し、2つのパラメーターを持つメソッドを公開するアダプタークラスを作成します。 このメソッドは、保存されたロガーに呼び出しを委任し、保存された「ソース」をロガー関数の最初のパラメーターとして渡します。 BusinessLogicクラスでは、アダプターのインスタンスを作成し、フィールドへのリンクを保存してから、2つのパラメーターを指定してメソッドを呼び出すだけです。 BusinessLogicのアダプターのみが必要な場合は、おそらくこれはやり過ぎですが、同じロギング機能を適応させる限り、再利用できます。
- BusinessLogicクラスにソースロガーオブジェクトを格納しますが、2つのパラメーターを持つヘルパーメソッドを作成します。その中には、「ソース」パラメーターの値がハードコーディングされます。 複数の場所でこれを行う必要がある場合、それは迷惑になり始めます。
- より機能的なアプローチを使用してください-この場合、 関数の部分的な使用 。
ロガーオブジェクトへのリンクを保存することと、ログ機能へのリンクを保存することの違いを意図的に無視します。 ロガークラスの複数の関数を使用する必要がある場合、明らかに大きな違いがありますが、カリー化と部分的なアプリケーションを考えるために、「ロガー」を「3つのパラメーターを取る関数」と考えます(例の関数として) )
動機付けのための擬似現実的な具体例を挙げたので、記事の終わりまでそれを忘れ、関数の例のみを検討します。 何か役に立つことをするふりをするBusinessLogicクラス全体を書きたくありません。 「例の関数」から「本当にやりたいこと」への精神的な変革ができると確信しています。
部分関数の使用
部分的なアプリケーションは、N個のパラメーターとこれらのパラメーターのいずれかの値を持つ関数を取り、N-1パラメーターを持つ関数を返します。そのため、呼び出されると、必要なすべての値(部分アプリケーション関数自体に渡される最初の引数と、残りのN-1戻り関数に渡される引数)。 したがって、これらの2つの呼び出しは、3つのパラメーターを持つメソッドと同等である必要があります。
この場合、1行目の最初の1つのパラメーターで部分的なアプリケーションを実装しました。ApplyPartial
を記述
できます。これは、より多くの引数を取るか、関数の最終実行の別の位置でそれらを置き換えます。 どうやら、最初からパラメータを1つずつ収集することが最も一般的なアプローチです。
無名関数(この場合はラムダ式ですが、匿名メソッドはそれほど冗長ではありません)のおかげで、ApplyPartialの実装は簡単です。
static Func<T2, T3, TResult> ApplyPartial<T1, T2, T3, TResult> (Func<T1, T2, T3, TResult> function, T1 arg1) { return (b, c) => function(arg1, b, c); }
一般化により、このメソッドは実際よりも複雑に見えます。 C#に高次型がないため、使用するデリゲート
ごとにこのメソッドを実装する必要があることに注意してください。4つのパラメーターを持つ関数のバージョンが必要な場合は、ApplyPartial <T1、T2メソッドが必要です。 、T3、T4、TResult>など。 また、おそらくアクションデリゲートファミリのメソッドのセットが必要です。
最後に注意することは、これらのすべてのメソッドを使用すると、必要に応じて、パラメーターなしの結果関数まで、部分的なアプリケーションを再度実行できることです。
Func<int, int, string> partial1 = ApplyPartial(function, 1); Func<int, string> partial2 = ApplyPartial(partial1, 2); Func<string> partial3 = ApplyPartial(partial2, 3); string result = partial3();
繰り返しますが、最後の行のみが元の関数を呼び出します。
わかりました、これは機能の部分的な適用です。
比較的簡単です。 私の意見では、カレーは理解するのが少し難しいです。
キャリング
部分的なアプリケーションは、1つの引数を使用して、N個のパラメーターを持つ関数をN-1個のパラメーターを持つ関数に変換しますが、カリー化は、関数を1つの引数の関数に分解します。 変換された関数を除いて、Curryメソッドに追加の引数を渡しません。
- カリー(f)は、次のような関数f1を返します。
- f1(a)は、次のような関数f2を返します。
- f2(b)は、次のような関数f3を返します...
- f3(c)はf(a、b、c)を引き起こします
(繰り返しますが、これは3つのパラメーターを持つ関数にのみ適用されることに注意してください。他の署名でどのように機能するかを明らかに期待しています。)
「同等」の例では、次のように記述できます。
最後の例の違いは、関数型言語がしばしば型推論と関数型のコンパクトな表現を持つことが多い理由を示しています。宣言f1は非常に怖いです。
Curryメソッドの処理内容がわかったので、その実装は驚くほど簡単です。 実際、上記のポイントをラムダ式に変換するだけです。 美しさ:
static Func<T1, Func<T2, Func<T3, TResult>>> Curry<T1, T2, T3, TResult> (Func<T1, T2, T3, TResult> function) { return a => b => c => function(a, b, c); }
コードをより理解しやすくしたい場合は、括弧を自由に追加してください。個人的には混乱を招くだけだと思います。 いずれにせよ、私たちは欲しいものを手に入れました。 (ラムダ式や匿名メソッドなしで書くのがどれだけ疲れるのかを考える価値があります。
難しくなく、ただ疲れます。)
これはカレーです。 そう思う。 おそらく。
おわりに
私はカレーを使用したことがあるとは言えませんが、
野田時間のテキスト解析の一部では実際に部分的なアプリケーションを使用しています。 (誰かが本当に私にこれをチェックして欲しいなら、私はそれをするでしょう。)
これら2つの関連する概念の違いを示したとは思いますが、それでも非常に異なる概念です。 終わりになったので、2つのパラメーターを持つ関数でそれらの違いがどのように表示されるかを考えてください。3つのパラメーターを使用することにした理由を理解してください:)
私の第六の感覚は、カリー化は学術的文脈において有用な概念であるが、実際には部分的な適用がより有用であることを教えてくれます。 ただし、これは関数型言語を最大限に使用しなかった人の6番目の感覚です。 F#の使用を開始する場合は、補完的な投稿を書くかもしれません。 今、私は経験豊富な読者がコメントで有用な考えを提供できることを願っています。