Виклик функції з рядка в C #


145

Я знаю, що в php ви можете телефонувати на зразок:

$function_name = 'hello';
$function_name();

function hello() { echo 'hello'; }

Чи можливо це в .Net?


2
Я бачив, що publicтакож повинен бути і ваш метод .
запоо

Відповіді:


268

Так. Можна використовувати рефлексію. Щось на зразок цього:

Type thisType = this.GetType();
MethodInfo theMethod = thisType.GetMethod(TheCommandString);
theMethod.Invoke(this, userParameters);

56
А для цього потрібно "використовувати System.Reflection;"
jptsetung

1
Чи можна це також використовувати для виклику функції з параметром: наприклад, рядок f = "метод (параметр1, параметр2)";
Грім

Дякую @ottobar за відповідь. Я не знаю, чи саме це я шукаю: мені потрібно було використовувати скалярну функцію SQL у своєму коді c #. Як я це називаю?
Чагбер

1
FYI це не працює, якщо ви перевантажили методи.
Шон О'Ніл

З наведеним вище кодом - метод, який викликається, повинен мати модифікатор доступу - PUBLIC. Якщо не є загальнодоступними, будь ласка, використовуйте Прапори прив'язки - BindingFlags.NonPublic | BindingFlags.Instance .. Type thisType = this.GetType(); MethodInfo theMethod = thisType.GetMethod(TheCommandString, BindingFlags.NonPublic | BindingFlags.Instance); theMethod.Invoke(this, userParameters);
Sibgath

75

Ви можете викликати методи екземпляра класу за допомогою відображення, роблячи виклик динамічного методу:

Припустимо, у вас є метод, який називається привіт у фактичному екземплярі (це):

string methodName = "hello";

//Get the method information using the method info class
 MethodInfo mi = this.GetType().GetMethod(methodName);

//Invoke the method
// (null- no parameter for the method call
// or you can pass the array of parameters...)
mi.Invoke(this, null);

1
як щодо методів класу "string"? використовуючи рамку 4
Леандро

36
class Program
    {
        static void Main(string[] args)
        {
            Type type = typeof(MyReflectionClass);
            MethodInfo method = type.GetMethod("MyMethod");
            MyReflectionClass c = new MyReflectionClass();
            string result = (string)method.Invoke(c, null);
            Console.WriteLine(result);

        }
    }

    public class MyReflectionClass
    {
        public string MyMethod()
        {
            return DateTime.Now.ToString();
        }
    }

це не залежно від класу
Леандро

А метод недійсний?
Кікенет

2

Незначна дотична - якщо ви хочете проаналізувати та оцінити цілий рядок виразів, який містить (вкладені!) Функції, врахуйте NCalc ( http://ncalc.codeplex.com/ та nuget)

Вих. трохи змінено з проектної документації:

// the expression to evaluate, e.g. from user input (like a calculator program, hint hint college students)
var exprStr = "10 + MyFunction(3, 6)";
Expression e = new Expression(exprString);

// tell it how to handle your custom function
e.EvaluateFunction += delegate(string name, FunctionArgs args) {
        if (name == "MyFunction")
            args.Result = (int)args.Parameters[0].Evaluate() + (int)args.Parameters[1].Evaluate();
    };

// confirm it worked
Debug.Assert(19 == e.Evaluate());

І всередині EvaluateFunctionделегата ви б назвали існуючу функцію.


0

У Факті я працюю над Windows Workflow 4.5, і мені вдалося знайти спосіб передати делегата від автомату до методу без успіху. Єдиний спосіб, який мені вдалося знайти, - це передати рядок з ім'ям методу, який я хотів передати як делегат і перетворити рядок у делегата всередині методу. Дуже приємна відповідь. Дякую. Перевірте це посилання https://msdn.microsoft.com/en-us/library/53cz7sc6(v=vs.110).aspx


0
Цей код працює в моїй консолі .Net
class Program
{
    static void Main(string[] args)
    {
        string method = args[0]; // get name method
        CallMethod(method);
    }
    
    public static void CallMethod(string method)
    {
        try
        {
            Type type = typeof(Program);
            MethodInfo methodInfo = type.GetMethod(method);
            methodInfo.Invoke(method, null);
        }
        catch(Exception ex)
        {
            Console.WriteLine("Error: " + ex.Message);
            Console.ReadKey();
        }
    }
    
    public static void Hello()
    {
        string a = "hello world!";
        Console.WriteLine(a);
        Console.ReadKey();
    }
}

Ласкаво просимо до SO! Мені подобається ваша відповідь, чи можете ви трохи більше пояснити, що ви робите?
a.deshpande012

Привіт, поза курсом. Це консольний додаток .net, всередині "програмного класу" є кілька методів чи функцій, наприклад: привіт, hello2, hello2. Пошук методу "CallMethod (строковий метод)", і дозволити мені викликати будь-який метод всередині "програмного класу", на який посилається ім'я як параметр рядка. Коли я виконую програму з "Консолі Windows": я пишу: "ім'я програми" + "метод мені потрібно викликати". Наприклад: myapp привіт, потім повернись "здрастуйте, світ!". Можливо також "myapp hello2" або "myapp hello3". Я думаю, може працювати в будь-якій програмі .Net.
ranobe

Ви повинні додати клас Reflection: "використовуючи System.Reflection;".
ranobe

-9

У C # ви можете створювати делегатів як покажчики функцій. Перегляньте наступну статтю MSDN щодо інформації про використання:http://msdn.microsoft.com/en-us/library/ms173171(VS.80).aspx

    public static void hello()
    {
        Console.Write("hello world");
    }

   /* code snipped */

    public delegate void functionPointer();

    functionPointer foo = hello;
    foo();  // Writes hello world to the console.

15
це не зі струн!
nawfal
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.