我正在尝试对DateTime进行限制的扩展方法?属性。这是一个搜索查询,我真的不想为所有可能的日期字段复制此代码。
public static IQueryOver<T, T> WhereInOpenEndedDateRange<T>(this IQueryOver<T, T> query,
Expression<Func<object>> field,
DateTime? rangeFrom,
DateTime? rangeTo)
{
if(rangeFrom.HasValue && rangeTo.HasValue)
{
query.WhereRestrictionOn(field).IsBetween(rangeFrom.Value).And(rangeTo.Value);
}
else if (rangeFrom.HasValue)
{
//query.Where(() => /* help */ >= rangeFrom.Value);
}
else if (rangeTo.HasValue)
{
//query.Where(() => /* help */ <= rangeTo.Value);
}
return query;
}
我想我错过了表达的一个基本方面。是否可以传递某种形式的Expression参数并在/* help */
点中使用它?
由于
更新
越来越近了,但仍然感觉很遥远......else if (rangeFrom.HasValue)
{
var lt = Expression.LessThanOrEqual(field, Expression.Constant(rangeFrom, typeof(DateTime?)));
var b = Expression.Lambda<Func<bool>>(lt);
query.Where(b);
}
但这不会有效,因为它会将Func<object>
与DateTime?
进行比较。如何将原始属性表达式合并到一个新的Func<bool>
中,并保留所需的位以保持NH QueryOver的顺畅?
在查看了QueryOverRestrictionBuilder的NH源代码后,我会做
string propertyName = ExpressionProcessor.FindMemberExpression(field.Body)
并使用Criteria方法构建限制。
答案 0 :(得分:2)
好吧,我无法弄清楚如何使用
来做到这一点Expression<Func<bool>>
签名,但这只适用于DateTime?所以我不知道我添加的限制将如何产生任何差异。
基本上,您需要表达式(示例中的字段)与您发送的参数完全匹配,即x =&gt; x.SomeNullableDateField。 'x'是您用
开始的查询类型session.QueryOver<YourClass>()
所以它需要包含在正在发送的表达式中。你也错过了使用
获取该参数ParameterExpression param = expression.Parameters.Single()
用于构造比较表达式。
您将BinaryExpression转换为Expression<Func<bool>>
,但您必须更进一步Expression<Func<T, bool>>
。
public static IQueryOver<T, T> WhereInOpenEndedDateRange<T>(this IQueryOver<T, T> query,
Expression<Func<T, DateTime?>> expression,
DateTime? rangeFrom,
DateTime? rangeTo) where T : class
{
// Lambda being sent in
ParameterExpression param = expression.Parameters.Single();
if(rangeFrom.HasValue && rangeTo.HasValue)
{
// GT Comparison
var expressionGT =
Expression.GreaterThanOrEqual(
expression.Body,
Expression.Constant(rangeFrom.Value, typeof(DateTime?)
)
);
// LT Comparison
var expressionLT =
Expression.LessThanOrEqual(
expression.Body,
Expression.Constant(rangeTo.Value, typeof(DateTime?)
)
);
query.Where(
Expression.Lambda<Func<T, bool>>(expressionGT, param))
.And(
Expression.Lambda<Func<T, bool>>(expressionLT, param)
);
}
else if(rangeFrom.HasValue)
{
// GT Comparison
BinaryExpression expressionGT =
Expression.GreaterThanOrEqual(
expression.Body,
Expression.Constant(rangeFrom.Value, typeof(DateTime?)
)
);
// covert to lambda
query.Where(Expression.Lambda<Func<T, bool>>(expressionGT, param));
}
else if(rangeTo.HasValue)
{
// LT Comparison
BinaryExpression expressionLT =
Expression.LessThanOrEqual(
expression.Body,
Expression.Constant(rangeTo.Value, typeof(DateTime?)
)
);
query.Where(Expression.Lambda<Func<T, bool>>(expressionLT, param));
}
return query;
}
用法
var test = session.QueryOver<MyPocoClass>()
.WhereInOpenEndedDateRange(x=>x.SomeNullableDateField, DateTime.Now, null);