我有一个必须过滤子属性的列表。过滤器运算符是动态的,我正在使用谓词构建器来组合多个过滤器/lambda。
为简单起见,假设我有两个这样的类:
public class FirstClass
{
public int Id { get; set; }
public ICollection<SecondClass> MyList { get; set; }
}
public class SecondClass
{
public int ReferenceId { get; set; }
public int Value { get; set; }
}
我的过滤器使用参考 id、运算符类型和值,这样伪代码将如下所示:
"list of FirstClass".Where(param1 =>
param1.MyList.Single(param2 =>
param2.ReferenceId == "reference id").Value "operatorType" "value")
实际的过滤器类似于123 eq 456
,其中参考 id 是 123,operatorType 是“eq”,值是 456。
如果运算符只是相等,那么以下工作就可以了:
Expression<Func<FirstClass, bool>> lambda =
param1 => param1.MyList.Single(param2 => param2.ReferenceId == id).Value == value;
此外,仅FirstClass
使用动态表达式进行过滤,就像一个魅力,例如过滤 Id(我的 ExpressionTypeDictionary 是一个字典,用于ExpressionType
根据提供的选择一个operatorType
):
var parameter = Expression.Parameter(typeof(FirstClass), "param1");
Expression body = parameter;
body = Expression.Property(body, "Id");
body = Expression.MakeBinary(ExpressionTypeDictionary[operatorType], body, value);
var lambda = Expression.Lambda<Func<FirstClass, bool>>(body, new[] { parameter });
我能够编译以下内容,但是使用 EF Core 对真实数据执行过滤器会返回 querySource 的异常:
var parameter = Expression.Parameter(typeof(FirstClass), "param1");
Expression<Func<FirstClass, int>> left = param1 =>
param1.MyClass.Single(param2 => param2.ReferenceId == id).Value;
var body = Expression.MakeBinary(
ExpressionTypeDictionary[operatorType],
left.Body,
Expression.Constant(value));
var lambda = Expression.Lambda<Func<FirstClass, bool>>(body, new[] { parameter });
...
theList.Where(lambda);
任何建议表示赞赏:)