web-dev-qa-db-ja.com

Entity Frameworkでwhere句を動的に追加する

私はこのSQLステートメントを持っています

SELECT userID from users WHERE
(name='name1' AND username='username1') OR
(name='name2' AND username='username2') OR
(name='name3' AND username='username3') OR
..........
(name='nameN' AND username='usernameN')

LINQを使用してエンティティフレームワークでこのステートメントを実装するにはどうすればよいですか?

27
Mironline

PredicateBuilder という美しいものを使用できます。このように使う

var pr = PredicateBuilder.False<User>();
foreach (var name in names)
{
    pr = pr.Or(x => x.Name == name && x.Username == name);
}
return query.AsExpandable().Where(pr);
31
Egor Pavlikhin
 Expression<Func<User, bool>> whereExpression = null;
 foreach (var name in names)
 {
     Expression<Func<User, bool>> e1 = u => u.Name == name;
     Expression<Func<User, bool>> andExpression = e1.And(u => u.Username == name);
     whereExpression = whereExpression == null ? andExpression : whereExpression.Or(andExpression);
 }
 return query.Where(whereExpression);

このヘルパーはあなたを助けるかもしれません。

public static class ExpressionExtensions
{
    public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> leftExpression, Expression<Func<T, bool>> rightExpression)
    {
        if (leftExpression == null) return rightExpression;
        if (rightExpression == null) return leftExpression;
        var paramExpr = Expression.Parameter(typeof(T));
        var exprBody = Expression.And(leftExpression.Body, rightExpression.Body);
        exprBody = (BinaryExpression)new ParameterReplacer(paramExpr).Visit(exprBody);

        return Expression.Lambda<Func<T, bool>>(exprBody, paramExpr);
    }

    public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> leftExpression, Expression<Func<T, bool>> rightExpression)
    {
        if (leftExpression == null) return rightExpression;
        if (rightExpression == null) return leftExpression;
        var paramExpr = Expression.Parameter(typeof(T));
        var exprBody = Expression.Or(leftExpression.Body, rightExpression.Body);
        exprBody = (BinaryExpression)new ParameterReplacer(paramExpr).Visit(exprBody);

        return Expression.Lambda<Func<T, bool>>(exprBody, paramExpr);
    }
}

class ParameterReplacer : ExpressionVisitor
{
    private readonly ParameterExpression _parameter;

    protected override Expression VisitParameter(ParameterExpression node)
    {
        return base.VisitParameter(_parameter);
    }

    internal ParameterReplacer(ParameterExpression parameter)
    {
        _parameter = parameter;
    }
}
8
Sergey Shuvalov

注:これは私が持っているものから変更されているため、そのままでは機能しない可能性があります。しかし、それは良い出発点になるでしょう。

    public static IQueryable<TEntity> Where<TEntity>(this IQueryable<TEntity> source,
                                                     IEnumerable<WhereSpecifier> orClauses)
        where TEntity : class
    {
        if (!orClauses.Any()) return source.Where(t => false);
        Type type = typeof (TEntity);
        ParameterExpression parameter = null;
        Expression predicate = Expression.Constant(false, typeof (bool));
        ParameterExpression whereEnt = Expression.Parameter(type, "WhereEnt");
        foreach (WhereSpecifier orClause in orClauses)
        {
            Expression selector;
            if (orClause.Selector != null)
            {
                selector = orClause.Selector;
                parameter = orClause.Parameter;
            }
            else
            {
                parameter = whereEnt;
                Type selectorResultType;
                selector = GenerateSelector<TEntity>(parameter, orClause.Column, out selectorResultType);
            }
            Expression clause = selector.CallMethod(orClause.Method, 
                MakeConstant(selector.Type, orClause.Value), orClause.Modifiers);
            predicate = Expression.Or(predicate, clause);
        }

        var lambda = Expression.Lambda(predicate, whereEnt);
        var resultExp = Expression.Call(typeof (Queryable), "Where", new[] {type},
            source.Expression, Expression.Quote(lambda));
        return source.Provider.CreateQuery<TEntity>(resultExp);
    }

GenerateSelector:

public static Expression GenerateSelector<TEntity>(ParameterExpression parameter, string propertyName,
                                                         out Type resultType) where TEntity : class
    {
        //  create the selector part, but support child properties
        PropertyInfo property;
        Expression propertyAccess;
        if (propertyName.Contains('.'))
        {
            // support to be sorted on child fields.
            String[] childProperties = propertyName.Split('.');
            property = typeof (TEntity).GetProperty(childProperties[0]);
            propertyAccess = Expression.MakeMemberAccess(parameter, property);
            for (int i = 1; i < childProperties.Length; i++)
            {
                property = property.PropertyType.GetProperty(childProperties[i]);
                propertyAccess = Expression.MakeMemberAccess(propertyAccess, property);
            }
        }
        else
        {
            property = typeof (TEntity).GetProperty(propertyName);
            propertyAccess = Expression.MakeMemberAccess(parameter, property);
        }
        resultType = property.PropertyType;
        return propertyAccess;
    }

WHereSpecifier:

public class WhereSpecifier
{
    public WhereSpecifier(string column, CheckMethod method, string value, CheckMethodModifiers modifiers)
    {
        Modifiers = modifiers;
        Value = value;
        Column = column;
        Method = method;
    }

    public WhereSpecifier(string column, CheckMethod method, string value)
        : this(column, method, value, CheckMethodModifiers.None)
    {
    }
    public Expression Selector { get; set; }
    public ParameterExpression Parameter { get; set; }
    public string Column { get; set; }
    public CheckMethod Method { get; set; }
    public CheckMethodModifiers Modifiers { get; set; }
    public string Value { get; set; }
}

使用法:

var column = typeof(TEntity).Name + "ID";
var where = from id in SelectedIds 
            select new WhereSpecifier(column, CheckMethod.Equal, id.ToString());
return GetTable().Where(where);
4
tster

@Egor Pavlikhinソリューションを試しましたが、"The LINQ expression node type 'Invoke' is not supported in LINQ to Entities."

this によると、 PredicateExtensions を使用できます。

var predicate = PredicateExtensions.Begin<User>();
foreach (var name in names)
{
    pr = pr.Or(x => x.Name == name);
}
return _context.Users.Where(predicate);
2
Medo Elkamaly

エンティティフレームワークは entity sql も理解するので、クエリのこの部分を文字列で実行できることを忘れないでください。文字列を作成することは、動的なことを行う必要がある場合に非常に便利です。

2
Giovanni Galbo

私はこのようにそれがあまりにも単純であることを発見しました:

    var query = context.InvoiceHeader.Where( i =>  i.DateInvoice >= model.datedu && i.DateInvoice <= model.dateau).AsQueryable();
        if(model.name != null)
        {
            query = query.Where(i =>  i.InvoiceNum.Equals(model.name));
        }
        if (model.status != 0 )
        {
            query = query.Where(i => i.REF_InvoiceStatusRecId == model.status);
        }

        if (model.paiements != 0)
        {
            query = query.Where(i => i.REF_PaymentTermRecId  == model.paiements);
        }
        query = query.AsQueryable().OrderByDescending(x => x.RecId);
0
saad kabarousse

ユーザーインターフェイスの選択に基づいて動的に 'Where'句の述語を作成する必要がありました。 'System.Dynamic.Linq'は、文字列からの述語を許可します。

foreach (var name in names)
{
    query = query.Where("Name=@0 And UserName=@1", name, name);
}
return query;

「System.Dynamic.Linq」は、nugetパッケージとして入手できます。 Scott Guthrieによるトピックの紹介 here をご覧ください。

0
Shyam Poovaiah