NH QueryOver扩展:如何将成员表达式合并到另一个表达式中?

时间:2011-04-15 17:24:25

标签: c# nhibernate expression queryover

我正在尝试对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方法构建限制。

1 个答案:

答案 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);