如何在运行时在IEnumerable <t>上创建动态选择?</t>

时间:2012-01-24 16:18:24

标签: c# linq reflection duck-typing

鉴于我有一个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 });
}

我将其保留为Expression,因为调用Compile会导致IQueryable被枚举,这意味着数据库被不必要地命中。因此,要使用它,我只需执行以下操作:

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

3 个答案:

答案 0 :(得分:6)

可以使用表达式

来完成此操作

<强> e.g。

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();

}

hth,
艾伦。

答案 1 :(得分:0)

动态linq库允许您动态指定谓词和投影,并且可能适合您的用例 -

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

答案 2 :(得分:0)

您可以动态构建Expression<Func<T, U>>