Expression.PropertyOrField为string属性返回“不是'System.String'类型的成员”

时间:2016-03-25 14:49:06

标签: c# .net lambda expression-trees

我正在尝试为规则引擎动态构建表达式,并且事情进展顺利,直到我尝试允许将嵌套类型和属性指定为操作数。样品:

的ExpressionBuilder

public Expression BuildExpression<T>(string propertyName, Enums.Operator ruleOperator, object value, ParameterExpression parameterExpression)
    {
        ExpressionType expressionType = new ExpressionType();
        Expression body = parameterExpression;

        foreach (var member in propertyName.Split('.'))
        {
            body = MemberExpression.Property(body, member);
        }

        var leftOperand = MemberExpression.PropertyOrField(body, propertyName);
        var rightOperand = Expression.Constant(Convert.ChangeType(value, value.GetType()));
        FieldInfo fieldInfo = expressionType.GetType().GetField(Enum.GetName(typeof(Enums.Operator), ruleOperator));
        var expressionTypeValue = (ExpressionType)fieldInfo.GetValue(ruleOperator);

        return CastBuildExpression(expressionTypeValue, value, leftOperand, rightOperand);
    }

RuleEngine

public Func<T, bool>[] CombineRules<T>(Criterion[] criteria)
    {
        List<Func<T, bool>> list = new List<Func<T, bool>>();
        foreach (var criterion in criteria)
        {                    
            ExpressionBuilder expressionBuilder = new ExpressionBuilder();
            var param = Expression.Parameter(typeof (T));
            Expression expression = expressionBuilder.BuildExpression<T>(criterion.PropertyName,
                    criterion.Operator_, criterion.Value, param);
            Func<T, bool> func = Expression.Lambda<Func<T, bool>>(expression, param).Compile();
            list.Add(func);
        }

        return list.ToArray();
    }

标准

public class Criterion
{
    private bool propertySet;
    public string PropertyName { get; set; }
    public Enums.Operator Operator_ { get; set; }
    public object Value { get; set; }

MemberModel

public class MemberModel
{
    public string UserName{ get; set; }
    public PersonalDetailsModel PersonalDetails {get; set;}
}

PersonalDetailsModel

    public class PersonalDetailsModel
{
    public int PersonalDetailsId { get; set; }
    public string Firstname { get; set; }
    public string Lastname { get; set; }
    public string Middlename { get; set; }
    public string DateOfBirth { get; set; }
    public string GenderType { get; set; }
    public string SalutationType { get; set; }
}

当我尝试将嵌套属性作为左操作数,即PropertyName =“PersonalDetails.FirstName”传递到ruleEngine.CombineRules(criteria.ToArray())时出现问题;

我得到“'PersonalDetails.Firstname'不是'System.String'类型的成员”,尽管它显然是这样。我现在已经坚持了一段时间,不知道可能导致它的原因吗?

非常感谢任何帮助。

1 个答案:

答案 0 :(得分:3)

下面

    Expression body = parameterExpression;

    foreach (var member in propertyName.Split('.'))
    {
        body = MemberExpression.Property(body, member);
    }

你已经处理了属性路径,所以这个

  var leftOperand = MemberExpression.PropertyOrField(body, propertyName);

毫无意义,是异常的来源。在您的示例中,body包含p.PersonalDetails.FirstName(字符串)之类的内容,上面一行正在尝试构建类似p.PersonalDetails.FirstName.PersonalDetails.FirstName的内容。

改为使用var leftOperand = body;

您可以使用简单的

缩短整个属性路径处理
var leftOperand = propertyName.Split('.')
    .Aggregate((Expression)parameterExpression, Expression.PropertyOrField);