ti-enxame.com

C # passando função como argumento

Eu escrevi uma função em C # que faz uma diferenciação numérica. Se parece com isso:

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

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

Eu gostaria de poder passar em qualquer função, como em:

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

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

Eu acho que isso é possível com os delegados (talvez?), Mas não tenho certeza de como usá-los.

Qualquer ajuda seria muito apreciada.

114
Ash

Usando o Func como mencionado acima funciona, mas também há delegados que fazem a mesma tarefa e também definem a intenção dentro da nomenclatura:

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);
}
123
Ian Johnson

Existem alguns tipos genéricos em .Net (v2 e posteriores) que facilitam a passagem de funções como delegados.

Para funções com tipos de retorno, há Func <> e para funções sem tipos de retorno existe Action <>.

Ambos Func e Action podem ser declarados para levar de 0 a 4 parâmetros. Por exemplo, Func <double, int> recebe um duplo como parâmetro e retorna um int. A ação <double, double, double> aceita três duplas como parâmetros e não retorna nada (void).

Então você pode declarar sua função Diff para obter uma Func:

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

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

E então você o chama assim, simplesmente dando a ele o nome da função que se ajusta à assinatura de sua Func ou Action:

double result = Diff(myValue, Function);

Você pode até escrever a função em linha com a sintaxe lambda:

double result = Diff(myValue, d => Math.Sqrt(d * 3.14));
152
quentin-starin
public static T Runner<T>(Func<T> funcToRun)
{
    //Do stuff before running function as normal
    return funcToRun();
}

Uso:

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