Expression.MemberInit - 最佳重载方法匹配具有一些无效参数

时间:2014-07-25 15:41:02

标签: c# .net linq

我有一个与以下代码相关的问题;基本上我想创建一个动态LINQ表达式,允许我在LINQ中创建一个基于字符串的Select语句:

public class SelectBuilder<T, T>
{
    Func<T, T> CreateNewStatement(string fields)
    {
        // input parameter "o"
        var xParameter = Expression.Parameter(typeof(T), "o");

        // new statement "new Data()"
        var xNew = Expression.New(typeof(T));

        // create initializers
        var bindings = fields.Split(',').Select(o => o.Trim())
            .Select(o =>
            {

                // property "Field1"
                var mi = typeof(T).GetProperty(o);

                // original value "o.Field1"
                var xOriginal = Expression.Property(xParameter, mi);

                // set value "Field1 = o.Field1"
                return Expression.Bind(mi, xOriginal);
            }
        );

        // initialization "new Data { Field1 = o.Field1, Field2 = o.Field2 }"
        var xInit = Expression.MemberInit(xNew, bindings); // <-- Error on this line

        // expression "o => new Data { Field1 = o.Field1, Field2 = o.Field2 }"
        var lambda = Expression.Lambda<Func<T, T>>(xInit, xParameter);

        // compile to Func<Data, Data>
        return lambda.Compile();
    }
}

这是LINQ表达式:

var vLINQ = (from a in PrimaryTable.AsEnumerable()
                                 join b in vDTsec.AsEnumerable()
                                 on a.Field<Object>("ID") equals b.Field<Object>("ID")
                                 into Group
                                 from q in Group.DefaultIfEmpty()
                                 select q).Select(MYSELECTSTRING);

这是错误: Error

编译器给出了Microsoft FW 3.5的问题;在FW 4.0或以上,我没有问题。但是我需要在3.5中编译项目。 有人知道这个问题的解决方法吗?

1 个答案:

答案 0 :(得分:2)

您遇到方差问题。您的bindings变量是IEnumerable<MemberAssignment>,其中MemberInit期待IEnumerable<MemberBinding>。由于T在.NET 4.0中的IEnumerable<out T>中是协变的,因此可以正常工作。为了更多地说明这个问题,发生了什么:

IEnumerable<MemberBinding> foo = bindings;

您可以通过强制转换绑定来轻松解决此问题:

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

                // property "Field1"
                var mi = typeof(TA).GetProperty(o);

                // original value "o.Field1"
                var xOriginal = Expression.Property(xParameter, mi);

                // set value "Field1 = o.Field1"
                return (MemberBinding)Expression.Bind(mi, xOriginal);
            }
        );

这将始终有效,因为MemberAssignment派生自MemberBinding。