引数として関数を渡すC#


141

数値微分を行う関数をC#で記述しました。次のようになります。

public double Diff(double x)
{
    double h = 0.0000001;

    return (Function(x + h) - Function(x)) / h;
}

次のように、任意の関数を渡すことができるようにしたいと思います。

public double Diff(double x, function f)
{
    double h = 0.0000001;

    return (f(x + h) - f(x)) / h;
}

これはデリゲートで可能だと思います(たぶん?)が、それらの使用方法はわかりません。

どんな助けでも大歓迎です。

回答:


146

上記のFuncの使用は機能しますが、同じタスクを実行し、ネーミング内で意図を定義するデリゲートもあります。

public delegate double MyFunction(double x);

public double Diff(double x, MyFunction f)
{
    double h = 0.0000001;

    return (f(x + h) - f(x)) / h;
}

public double MyFunctionMethod(double x)
{
    // Can add more complicated logic here
    return x + 10;
}

public void Client()
{
    double result = Diff(1.234, x => x * 456.1234);
    double secondResult = Diff(2.345, MyFunctionMethod);
}

5
3.5以降では、Func <>とデリゲートは交換可能です。つまり、匿名デリゲートとラムダ(匿名デリゲートの構文シュガー)も使用できます。したがって、パラメーターをFunc <double、double>として指定するか、doubleを受け取ってdoubleを返すデリゲートとして指定するかは、実際には重要ではありません。名前付きデリゲートがあなたに与える唯一の真の利点は、xml-docコメントを追加できることです。わかりやすい名前は、型の代わりにパラメーター名と同じくらい簡単に実装できます。
KeithS

3
メソッドプロトタイプはFunc <x、y>よりもコードを読みやすくしていると私は主張します。コードを読みやすくするのは名前だけではありません。
Ian Johnson、

デリゲート型の命名がコードの明快さにとって非常に重要である場合、私はほとんどの場合、インターフェイスと実装に頼っていると思います。
クエンティン・スターリン

@qstarinこれは、デリゲートの名前だけでなく、メソッドが取る必要のある引数の名前でもあります。特に、それらがネイティブ型の場合はそうです。あなたは正しい、ほとんどの場合、私はデリゲートを介してインターフェイスを使用する
Ian Johnson

これらのいずれかを、一般的な機能を提供する一般的な関数として(あまり一般的でない戻り値の型で)作成している場合、voidで使用しようとするまで、すべてが機能します。Funcの代わりにActionを使用して関数を複製する必要があるのですか、それとももっと良い方法がありますか?
Dan Chase

175

.Net(v2以降)には、デリゲートとしての関数の受け渡しを非常に簡単にするいくつかのジェネリック型があります。

戻り型のある関数にはFunc <>があり、戻り型のない関数にはAction <>があります。

FuncとActionはどちらも、0から4つのパラメーターを取るように宣言できます。たとえば、Func <double、int>は1つのdoubleをパラメーターとして取り、intを返します。アクション<double、double、double>は、パラメーターとして3つのdoubleを取り、何も返しません(void)。

したがって、Diff関数を宣言してFuncを取得できます。

public double Diff(double x, Func<double, double> f) {
    double h = 0.0000001;

    return (f(x + h) - f(x)) / h;
}

そして、それをそのまま呼び出し、FuncまたはActionのシグネチャに適合する関数の名前を付けます。

double result = Diff(myValue, Function);

ラムダ構文を使用して関数をインラインで記述することもできます。

double result = Diff(myValue, d => Math.Sqrt(d * 3.14));

29
.NET 4では、Funcおよびの両方Actionが更新され、最大16個のパラメーターを使用できるようになりまし
Joel Mueller

5
Func<double, double>もちろん、本当に素晴らしいことは、入力関数の最初の導関数であるaを返すことです。もちろん、数値的に計算されます。入力関数の3次導関数return x => (f(x + h) - f(x)) / h;を返すオーバーロードを作成することもできnます。
Ani

15
public static T Runner<T>(Func<T> funcToRun)
{
    //Do stuff before running function as normal
    return funcToRun();
}

使用法:

var ReturnValue = Runner(() => GetUser(99));

気になるのですが、なぜジェネリック型パラメーターですか?
kdbanman 2015

2
このエラーが発生します。メソッド 'Runner <T>(Func <T>)'の型引数は、使用法から推測できません。型引数を明示的に指定してみてください。
DermFrench

「funcToRun」メソッドのシグネチャは何ですか?
kravits88 2015
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.