Как я могу создать динамический Select для IEnumerable <T>во время выполнения? - PullRequest
7 голосов
/ 24 января 2012

Учитывая, что у меня есть IEnumerable<T>, где T - это любой объект, как я могу выбрать из него определенное свойство, учитывая, что я знаю имя одного из имен свойств во время выполнения в виде строки?

Например:

var externalIEnumerable = DataPassedFromConsumingCode(); // `IEnumerable<T>`

string knownPropertyName = "Foo";
var fooSelect = externalIEnumerable.Select(...);

По сути, я, очевидно, просто делаю externalIEnumerable.Select(x=> x.Foo);, но мне нужно выполнить это Select во время выполнения, когда у меня нет контроля над тем, когда он изначально создан.

-

ОТВЕТ: Основываясь на ответе AlanT, вот что я на самом деле сделал:

public Expression<Func<TItem, object>> SelectExpression<TItem>(string fieldName)
{
    var param = Expression.Parameter(typeof(TItem), "item");
    var field = Expression.Property(param, fieldName);
    return Expression.Lambda<Func<TItem, object>>(field, 
        new ParameterExpression[] { param });
}

Я сохранил его как выражение, потому что вызов Compile вызвал перечисление IQueryable, что означало, что к базе данных был применен ненужный результат. Итак, чтобы использовать его, я просто делаю следующее:

string primaryKey = _map.GetPrimaryKeys(typeof(TOriginator)).Single();
var primaryKeyExpression = SelectExpression<TOriginator>(primaryKey);
var primaryKeyResults = query.Select(primaryKeyExpression).ToList();

Ответы [ 3 ]

6 голосов
/ 24 января 2012

Это можно сделать с помощью выражения

, например

private class Foo {
    public string Bar { get; set; }
}

private IEnumerable<Foo> SomeFoos = new List<Foo>() {
    new Foo{Bar = "Jan"},
    new Foo{Bar = "Feb"},
    new Foo{Bar = "Mar"},
    new Foo{Bar = "Apr"},
};

[TestMethod]
public void GetDynamicProperty() {

        var expr = SelectExpression<Foo, string>("Bar");
        var propValues = SomeFoos.Select(expr);

        Assert.IsTrue(new[] { "Jan", "Feb", "Mar", "Apr" }.SequenceEqual(propValues));

    }

public static Func<TItem, TField> SelectExpression<TItem, TField>(string fieldName) {

    var param = Expression.Parameter(typeof(TItem), "item");
    var field = Expression.Property(param, fieldName);
    return Expression.Lambda<Func<TItem, TField>>(field, new ParameterExpression[] { param }).Compile();

}

чч,
Алан.

0 голосов
/ 24 января 2012

Вы можете динамически построить Expression<Func<T, U>>.

0 голосов
/ 24 января 2012

Динамическая библиотека linq позволяет вам определять предикаты и проекции на лету и может соответствовать вашему сценарию использования -

http://weblogs.asp.net/scottgu/archive/2008/01/07/dynamic-linq-part-1-using-the-linq-dynamic-query-library.aspx

...