我正在尝试为规则引擎动态构建表达式,并且事情进展顺利,直到我尝试允许将嵌套类型和属性指定为操作数。样品:
的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'类型的成员”,尽管它显然是这样。我现在已经坚持了一段时间,不知道可能导致它的原因吗?
非常感谢任何帮助。
答案 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);