Как вызвать делегат из строки в C #? - PullRequest
2 голосов
/ 15 мая 2011

можно ли вызвать делегат, хранящийся в переменной, по имени переменной (в виде строки)? Я думаю, мне придется использовать механизм отражения, но я никуда не денусь

пример кода:

class Demo {
  public delegate int DemoDelegate();

  private static int One() {
    return 1;
  }
  private static void CallDelegate(string name) {
    // somehow get the value of the variable with the name
    // stored in "name" and call the delegate using reflection
  }
  private static void CallDelegate(string name, DemoDelegate d) {
    d();
  }
  static void main(string[] args) {
    DemoDelegate one = Demo.One;
    CallDelegate(one);
    // this works, but i want to avoid writing the name of the variable/delegate twice:
    CallDelegate("one", one);
  }

}

это вообще возможно? если так как? если нет, то я должен использовать вторую форму, невезение

Ответы [ 4 ]

5 голосов
/ 15 мая 2011

Переменные едва существуют.Единственный способ надежного вызова по строке (в этом сценарии) - хранить делегатов в словаре:

Dictionary<string, DemoDelegate> calls = new Dictionary<string, DemoDelegate>
{
    {"one",one}, {"two",two}
}

Теперь сохраните этот словарь где-нибудь ( в поле *).1008 *, как правило), и сделайте что-то вроде:

private int CallDelegate(string name) {
    return calls[name].Invoke(); // <==== args there if needed
}
3 голосов
/ 15 мая 2011

Да, это возможно, если вы используете выражения Linq и мало размышлений.

Посмотрите на этот код, он делает что-то похожее на то, что я думаю, вы хотите:

using System;
using System.Linq.Expressions;
using System.Reflection;
using System.Collections.Generic;

namespace q6010555
{
    class Demo
    {
        static List<string> varNamesUsed = new List<string>();

        public delegate int DemoDelegate();

        private static int One()
        {
            return 1;
        }
        private static void CallDelegate(Expression<Func<DemoDelegate>> expr)
        {
            var lambda = expr as LambdaExpression;
            var body = lambda.Body;
            var field = body as MemberExpression;
            var name = field.Member.Name;
            var constant = field.Expression as ConstantExpression;
            var value = (DemoDelegate)((field.Member as FieldInfo).GetValue(constant.Value));

            // now you have the variable name... you may use it somehow!
            // You could log the variable name.
            varNamesUsed.Add(name);

            value();
        }
        static void Main(string[] args)
        {
            DemoDelegate one = Demo.One;
            CallDelegate(() => one);

            // show used variable names
            foreach (var item in varNamesUsed)
                Console.WriteLine(item);
            Console.ReadKey();
        }
    }
}
1 голос
/ 15 мая 2011
public void Fire(string name)
        {
            FieldInfo field = this.GetType().GetField(name, BindingFlags.Public | BindingFlags.NonPublic | BindingFlags.Instance);

            if (field != null)
            {
                Delegate method = field.GetValue(this) as Delegate;

                if (method != null)
                {
                    method.Method.Invoke(method.Target, new object[0]);
                }
            }
        }

Очевидно ограничивает вас от параметризованных делегатов.

0 голосов
/ 15 мая 2011

Вы не можете реально получить доступ к переменным в другом фрейме стека (хотя я думаю, что это возможно, используя хакерство вокруг класса StackFrame). Вместо этого вы захотите передать объект Delegate и использовать методы, такие как DynamicInvoke, если вы хотите вызывать обобщенный делегат способом, похожим на отражение.

...