Сравните со строковыми значениями, используя IQueryable <T>и выражения - PullRequest
1 голос
/ 23 февраля 2011

Извините за название, но я не могу придумать, как быстро сказать, что я хочу - может кто-нибудь изменить его на более подходящий, если вы думаете о нем?

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

List.Compare("propretyName", ">1000");


public static IQueryable<T> Compare<T>(this IQueryable<T> source, string propertyName, string value) {
    Type type = typeof(T);
    ParameterExpression parameter = Expression.Parameter(type, "param");
    MemberExpression memberAccess = Expression.MakeMemberAccess(parameter, type.GetProperty(propertyName));

    string comparisonType = value.Substring(0, 1);
    value = value.Length > 0 ? value.Substring(1) : "0";

    decimal tmpvalue;
    decimal? result = decimal.TryParse(value, out tmpvalue) ? tmpvalue : (decimal?)null;
    ConstantExpression constant = Expression.Constant(result, typeof(decimal?));
    BinaryExpression comparisonExpression = Expression.GreaterThan(memberAccess, constant);
    switch (comparisonType) {
        case ">":
            comparisonExpression = Expression.GreaterThan(memberAccess, constant);
            break;
        case "<":
            comparisonExpression = Expression.LessThan(memberAccess, constant);
            break;
        case "=":
            comparisonExpression = Expression.Equal(memberAccess, constant);
            break;
    }

    Expression<Func<T, bool>> lambda = Expression.Lambda<Func<T, bool>>(comparisonExpression, parameter);
    return source.Where(lambda);
}

Выше приведен метод, который я написал для этого вызова.

Лямбда внизу кажется правильной: {param => (param.propretyName > 1000)}

Тем не менее, это не работает, и я думаю, что это потому, что конкретное свойство, над которым я работаю, это decimal?, поэтому оно должно быть {param => (param.propertyName.Value > 1000)}.

Может ли кто-нибудь помочь мне помочьиспользуйте значение вместо.Здесь что-то ускользает от меня.

Я не могу использовать метод Where(string), поскольку использую Entity-Framework.

Ответ найден

public static IQueryable<T> Compare<T>(this IQueryable<T> source, string propertyName, string value) {
    Type type = typeof(T);
    ParameterExpression parameter = Expression.Parameter(type, "param");
    MemberExpression memberAccess = Expression.MakeMemberAccess(parameter, type.GetProperty(propertyName));

    //This is the added methods that results in the proper output
    PropertyInfo valProp = typeof(Nullable<decimal>).GetProperty("Value");
    memberAccess = Expression.MakeMemberAccess(memberAccess, valProp);

    string comparisonType = value.Substring(0, 1);
    value = value.Length > 0 ? value.Substring(1) : "0";

    decimal tmpvalue;
    decimal? result = decimal.TryParse(value, out tmpvalue) ? tmpvalue : (decimal?)null;
    ConstantExpression constant = Expression.Constant(tmpvalue);
    BinaryExpression comparisonExpression = Expression.GreaterThan(memberAccess, constant);
    switch (comparisonType) {
        case ">":
            comparisonExpression = Expression.GreaterThan(memberAccess, constant);
            break;
        case "<":
            comparisonExpression = Expression.LessThan(memberAccess, constant);
            break;
        case "=":
            comparisonExpression = Expression.Equal(memberAccess, constant);
            break;
    }

    Expression<Func<T, bool>> lambda = Expression.Lambda<Func<T, bool>>(comparisonExpression, parameter);
    return source.Where(lambda);
}

1 Ответ

1 голос
/ 23 февраля 2011

Почему бы просто не использовать существующую Динамическую библиотеку LINQ :

myList.Where("propertyName > 1000");
...