Построить динамический запрос по универсальному - PullRequest
0 голосов
/ 17 января 2019

Я создал статический класс и попытался реализовать динамический запрос с общими параметрами. Ниже мое определение интерфейса и определение класса:

public interface IRangeValue<T> where T : struct
{
    Nullable<T> High{get;set;}
    Nullable<T> Low { get; set; }
}

public class DataRangeT<T> : IRangeValue<T>
    where T : struct
{
    Nullable<T> _high;
    Nullable<T> _low;
    public Nullable<T> High { get { return _high; } set { _high = value; } }
    public Nullable<T> Low { get { return _low; } set { _low = value; } }

}

Затем я создал статический метод для реализации динамического запроса:

public static class ExpressionHelperT<K>
    where K : struct
{
    public static Expression<Func<T, bool>> RangeCompare<T>(Expression<Func<T, K>> selector, IRangeValue<K> patten)
    {
        Expression<Func<T, bool>> predicate = PredicateBuilder.True<T>();
        if (patten.High.HasValue)
        {
            predicate = predicate.And<T>(Expression.Lambda<Func<T, bool>>(Expression.LessThanOrEqual(selector, Expression.Constant(patten.High.Value, typeof(K)))));
        }
        if (patten.Low.HasValue)
        {
            predicate = predicate.And<T>(Expression.Lambda<Func<T, bool>>(Expression.GreaterThanOrEqual(selector, Expression.Constant(patten.Low.Value, typeof(K)))));
        }
        return predicate;
    }

}

метод А определяется следующим образом:

public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> expr1,Expression<Func<T, bool>> expr2)
    {
        var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>());
        return Expression.Lambda<Func<T, bool>>(Expression.AndAlso(expr1.Body, invokedExpr), expr1.Parameters);
    }

Теперь я пытаюсь проверить мой метод RangeCompare<T>, я вызываю метод, как показано ниже:

class Program
{
    static void Main(string[] args)
    {
        DataRangeT<DateTime> dataRange = new DataRangeT<DateTime>();

        dataRange.High = null;
        dataRange.Low = DateTime.Today;

        List<DateTime> dates = new List<DateTime>();
        dates.Add(new DateTime(2018, 1, 1));
        dates.Add(new DateTime(2018, 2, 1));
        dates.Add(new DateTime(2018, 3, 1));
        dates.Add(new DateTime(2018, 4, 1));
        dates.Add(new DateTime(2018, 5, 1));
        dates.Add(new DateTime(2018, 6, 1));
        dates.Add(new DateTime(2018, 7, 1));
        dates.Add(new DateTime(2018, 8, 1));
        dates.Add(new DateTime(2018, 9, 1));

        List<DateTime> results = dates.Where<DateTime>(ExpressionHelperT<DateTime>.RangeCompare<DateTime>(c => c.Date, dataRange)).ToList();

        foreach (DateTime dt in results)
        {
            Console.WriteLine(dt.ToShortDateString());
        }
        Console.ReadLine();
    }
}

Но составитель всегда говорил мне:

Can't convert System.Collections.Generic.List<System.DateTime>”to“System.Linq.ParallelQuery<System.DateTime>”
Can't convert“System.Linq.Expressions.Expression<System.Func<System.DateTime,bool>>”to“System.Func<System.DateTime,int,bool>”

Почему?

1 Ответ

0 голосов
/ 17 января 2019

Ваше первое исключение не имеет ничего общего с кодом, который вы разместили

Не удалось преобразовать System.Collections.Generic.List <> »в« System.Linq.ParallelQuery <> »

Ваше второе исключение говорит само за себя

IEnumerbale не имеет такой сигнатуры метода

Where<TSource>(this IEnumerable<TSource> source, Expression<Func<TSource, bool>> predicate)

Проверьте себя Enumerable.Where Method

Если вы хотите, чтобы ваш Where поддерживал выражение , вам нужно будет преобразовать его в IQueryable, что, см. здесь

IQueryable<TSource> Where<TSource>(this IQueryable<TSource> source, Expression<Func<TSource, bool>> predicate)

Решить

Использовать AsQueryable Метод расширения для преобразования в IQueryable

var results = dates.AsQueryable()
                   .Where(ExpressionHelperT<DateTime>.RangeCompare<DateTime>(c => c.Date, dataRange)).ToList();
...