使用表达式构建器进行选择的动态lambda

Kog*_*oth 4 c# linq lambda

我正在尝试编写动态选择语句。我有以下几点:

public class MainList
{
    public string Prop1{ get; set; }
    public string Prop2{ get; set; }
    public string Prop3{ get; set; }       
}

public class SearchObject
{
    public string Prop1{ get; set; }
}
Run Code Online (Sandbox Code Playgroud)

我想构建如下表达式

var newList = MainList.Select(n => new SearchObject { Prop1 = n.Prop1});
Run Code Online (Sandbox Code Playgroud)

我使用的代码基于MainList创建一个列表。然后,我现在通过传递SearchObject类型和我想填充的参数来创建选择表达式。它一直运行到倒数第二行。

public void Start()
{
    List<MainList> newList = new List<MainList>(); //This has a ton list objects
    var result = newList.Select(CreateSelect<SearchObject>("Prop1"));
}

public static Func<MainList, T> CreateSelect<T>(string fields)
{
    var par = Expression.Parameter(typeof(T), "n");

    var newInstance= Expression.New(typeof(T));

    var bindings = fields.Split(',').Select(o => o.Trim())
        .Select(n => {

            var p = typeof(T).GetProperty(n);

            var original = Expression.Property(par, p);

            return Expression.Bind(p, original);
        }
    );

    var newT= Expression.MemberInit(newInstance, bindings);

    var lambda = Expression.Lambda<Func<MainList, T>>(newT, par); //ERROR HAPPENS HERE
    return lambda.Compile();
}
Run Code Online (Sandbox Code Playgroud)

我得到的错误是:

附加信息:类型'WebApplication.SearchObject'的ParameterExpression不能用于类型'WebApplication.MainList'的委托参数

我不确定错误的含义以及如何解决该问题。

Iva*_*oev 5

正如Jeroen van Langen已经提到的那样,第一个问题是参数的类型必须为MainList

第二个问题是的用法Expression.Bind。由于源和目标是不同的类型,因此不能使用相同的PropertyInfo。第一个参数必须是PropertyInfotarget类型的T,而第二个-表达式来自源类型MainList(在您的情况下,Expression.Property具有指定属性名称的参数上)。

正确的实现是这样的:

public static Func<MainList, T> CreateSelect<T>(string fields)
{
    var parameter = Expression.Parameter(typeof(MainList), "n");
    var bindings = fields.Split(',')
        .Select(name => name.Trim())
        .Select(name => Expression.Bind(
            typeof(T).GetProperty(name),
            Expression.Property(parameter, name)
        ));
    var newT = Expression.MemberInit(Expression.New(typeof(T)), bindings);
    var lambda = Expression.Lambda<Func<MainList, T>>(newT, parameter);
    return lambda.Compile();
}
Run Code Online (Sandbox Code Playgroud)