Передача дерева выражений в качестве параметра другому дереву выражений - PullRequest
7 голосов
/ 19 августа 2010

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

private Expression<Func<TEntity, TPropertyResult>> PropertyAccessor { get; set; }

и

private Expression<Func<TPropertyResult, bool>> TestExpression { get; set; }

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

var expression = p => this.TestExpression(this.PropertyAccessor(p));

При использовании Expression.Invoke(this.TestExpression, this.PropertyAccessor) я получаю следующую ошибку

{"Выражение типа 'System.Func`2 [MyEntity, System.String] нельзя использовать для параметра типа 'System.String' "}

TPropertyResult - строка во время моего теста.

Я пытался использовать Expression.Call или Expression.Invoke. Неудачно. Что я должен использовать?

1 Ответ

8 голосов
/ 19 августа 2010

Я думаю, что это то, о чем вы просите:

Expression<Func<TEntity, bool>> Combined
{
    get
    {
        var entity = Expression.Parameter(typeof(TEntity));
        var pa = Expression.Invoke(PropertyAccessor, entity);
        var te = Expression.Invoke(TestExpression, pa);
        return (Expression<Func<TEntity, bool>>) Expression.Lambda(te, entity);
    }
}

Я проверил это, и оно работает так, как я ожидал.

Однако, перечитайте ваш первоначальный вопрос (до того, какправки), у меня начинает складываться впечатление, что вы задали неправильный вопрос и что вам, вероятно, не нужны деревья выражений.Если все, что вам нужно, это функции , то вы можете использовать их без Expression:

private Func<TEntity, TPropertyResult> PropertyAccessor { get; set; }
private Func<TPropertyResult, bool> TestExpression { get; set; }
private Func<TEntity, bool> Combined
{
    get
    {
        return entity => TestExpression(PropertyAccessor(entity));
    }
}

Пример использования:

// Set up the original functions
PropertyAccessor = entity => GenerateResult(entity);
TestExpression = result => result.IsCool();

// This stores a reference to the combined function
var fn = Combined;

// This actually evaluates the function
bool isCool = fn(myEntity);

// Alternatively, you could evaluate the function directly, without the variable
bool isCool = Combined(myEntity);
...