Делегат по действиям. Как получить экземпляр, вызывающий метод

У меня есть действие, и мне интересно, как я могу получить доступ к экземпляру, вызывающему метод.

Exemple:

this.FindInstance(() => this.InstanceOfAClass.Method());
this.FindInstance(() => this.InstanceOfAClass2.Method());
this.FindInstance(() => this.InstanceOfAClass3.Method());



    public void FindInstance(Action action)
    {
        // The action is this.InstanceOfAClass.Method(); and I want to get the "Instance"
        // from "action"
    }

Спасибо

Ответ 1

Я думаю, что вы ищете свойство Delegate.Target.

EDIT: Хорошо, теперь я вижу, что вам нужно, и вам нужно дерево выражений, представляющее действие. Затем вы можете найти цель вызова метода в качестве другого дерева выражений, создать из этого выражения LambdaExpression, скомпилировать и выполнить его и увидеть результат:

using System;
using System.Linq.Expressions;

class Test
{
    static string someValue;

    static void Main()
    {
        someValue = "target value";

        DisplayCallTarget(() => someValue.Replace("x", "y"));
    }

    static void DisplayCallTarget(Expression<Action> action)
    {
        // TODO: *Lots* of validation
        MethodCallExpression call = (MethodCallExpression) action.Body;

        LambdaExpression targetOnly = Expression.Lambda(call.Object, null);
        Delegate compiled = targetOnly.Compile();
        object result = compiled.DynamicInvoke(null);
        Console.WriteLine(result);
    }
}

Обратите внимание, что это невероятно хрупкое, но оно должно работать в простых случаях.

Ответ 2

На самом деле я не знаю, сможешь ли ты сделать это таким образом. Delegate класс содержит только два свойства: Target и Method. Доступ к Target не будет работать, потому что вы создаете новый анонимный метод, поэтому свойство вернет класс, в котором вызывается метод FindInstance.

Попробуйте что-то вроде этого:

FindInstance(this.MyInstance.DoSomething);

И затем войдите в свойство Target следующим образом:

public void FindInstance(Action action)
{
    dynamic instance = action.Target;
    Console.WriteLine(instance.Property1);
}