我正在尝试创建一个用于为Entity Framework(5)撰写查询的泛型类。
我让它工作,唯一的问题是该值是作为查询的常量而不是作为参数注入的。这减少了EF缓存查询并在以后重用它的可能性。
这是我到目前为止所得到的。
public class MinDateFilter<T> : IFilter<T> where T : class
{
private readonly Expression<Func<T, bool>> _predicate;
public MinDateCandidateFilter(Expression<Func<T, DateTime>> propertySelector, DateTime from)
{
from = from.Date.AddDays(-1);
from = new DateTime(from.Year, from.Month, from.Day, 23, 59, 59, 999);
Expression value = Expression.Constant(from, typeof(DateTime));
//ParameterExpression variable = Expression.Variable(typeof(DateTime), "value");
MemberExpression memberExpression = (MemberExpression)propertySelector.Body;
ParameterExpression parameter = Expression.Parameter(typeof(T), "item");
Expression exp = Expression.MakeMemberAccess(parameter, memberExpression.Member);
Expression operation = Expression.GreaterThan(exp, value);
//Expression operation = Expression.GreaterThan(exp, variable);
_predicate = Expression.Lambda<Func<T, bool>>(operation, parameter);
}
public IQueryable<T> Filter(IQueryable<T> items)
{
return items.Where(_predicate);
}
}
这个类可以用两种方式:
通过对它进行细分:
public class MinCreationDateCandidateFilter : MinDateFilter<Candidate>
{
public MinCreationDateCandidateFilter(DateTime @from) : base(c => c.CreationDate, @from) {}
}
或仅仅通过实例化它:
var filter = new MinDateFilter<Entities.Transition>(t => t.Date, from.Value);
这是我到目前为止所取得的成就:
SELECT
[Extent1].[Id] AS [Id]
-- Other fields
FROM [dbo].[Candidates] AS [Extent1]
WHERE [Extent1].[CreationDate] > convert(datetime2, '1982-12-09 23:59:59.9990000', 121)
而不是
SELECT
[Extent1].[Id] AS [Id]
-- Other fields
FROM [dbo].[Candidates] AS [Extent1]
WHERE [Extent1].[CreationDate] > @p__linq__0
如果我取消注释两条注释行并对上面两条注释,我会收到一条错误,指出参数“value”没有绑定。
我希望我提供了所有有用的细节:)
答案 0 :(得分:11)
当参数作为ConstantExpression
传递时,如下所示:
Expression.Constant(myString)
...它将在生成的查询中生成一个固定的常量符号:
SELECT
[Extent1].[Id] AS [Id],
[Extent1].[Bar] AS [Bar],
FROM [dbo].[Foo] AS [Extent1]
WHERE [Extent1].[Bar] = "Some text"
如果您像使用Expression Tree Visualizer一样使用某种工具来分析像(f => f.Bar == myString))
这样的表达式,您会看到该参数实际上是MemberExpression
。
因此,如果您想在结果查询中使用参数,则必须传递类似对象属性或更方便的匿名类型的内容:
Expression.Property(
Expression.Constant(new { Value = myString }),
"Value"
)
通过这种方式,您传递刚刚创建的对象的属性,表达式树获得MemberExpression
,从而生成以下CommandText
:
SELECT
[Extent1].[Id] AS [Id],
[Extent1].[Bar] AS [Bar],
FROM [dbo].[Foo] AS [Extent1]
WHERE [Extent1].[Bar] = @p__linq__0