Как создать пустой делегат, используя деревья выражений? - PullRequest
6 голосов
/ 22 марта 2012

Используя анонимные методы , вы можете создавать пустые делегаты начиная с C # 2.0.

public event EventHandler SomeEvent = delegate {};
public event Action OtherEvent = delegate {};

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

Как создать такое же поведение, используя Деревья выражений ?

Единственный возможный вариант, который я вижу сейчас, - это использовать Expression.Lambda(), но, насколько я могу судить, для этого потребуется много дополнительной работы.

Ответы [ 3 ]

5 голосов
/ 22 марта 2012

Дерево выражений по своей природе всегда имеет выражение для тела, а не выражение в исходном проекте.

В C # 3 вообще не было способа выразить дерево выражений, чьеТело - это пустой блок операторов.Совсем недавно библиотека дерева выражений была расширена для поддержки операторов, но правила семантического анализа C # не были обновлены, чтобы воспользоваться этим;вы все еще не можете превратить лямбда-выражение в дерево выражений.

2 голосов
/ 22 марта 2012

Как выясняется, это не , что много работает с использованием Expression.Lambda().Однако меня все еще интересуют возможные другие ответы.

Мне нужен вспомогательный метод, который я написал ранее:

/// <summary>
///   The name of the Invoke method of a Delegate.
/// </summary>
const string InvokeMethod = "Invoke";

/// <summary>
///   Get method info for a specified delegate type.
/// </summary>
/// <param name = "delegateType">The delegate type to get info for.</param>
/// <returns>The method info for the given delegate type.</returns>
public static MethodInfo MethodInfoFromDelegateType( Type delegateType )
{
    Contract.Requires(
        delegateType.IsSubclassOf( typeof( MulticastDelegate ) ),
        "Given type should be a delegate." );

    return delegateType.GetMethod( InvokeMethod );
}

Когда у вас есть EventInfoвы можете создать для него пустую лямбду следующим образом:

EventInfo _event;

...

MethodInfo delegateInfo
    = DelegateHelper.MethodInfoFromDelegateType( _event.EventHandlerType );
ParameterExpression[] parameters = delegateInfo
    .GetParameters()
    .Select( p => Expression.Parameter( p.ParameterType ) )
    .ToArray();
Delegate emptyDelegate = Expression.Lambda(
    _event.EventHandlerType,
    Expression.Empty(), "EmptyDelegate", true, parameters ).Compile();
1 голос
/ 13 марта 2017

Немного расширив ответ Стивена - мне потребовались аналогичные функциональные возможности для создания пустого делегата для обоих типов - Action и Func - вот вспомогательный инструмент, который я создал для этой задачи:

...