Използвайте PredicateBuilder в SelectMany LINQ to Entity Framework

За първи път пиша тук в stackoverflow.

Бих искал да използвам PredicateBuilder в LINQ заявка с SelectMany.

Въвеждам кода си

public async Task<List<Articolo>> OttieniElencoArticoliFiltratoComplessoAsync
    (ExpressionStarter<Articolo> predicateArticolo,
        ExpressionStarter<ArticoloFornitore> predicateFornitore,
        ExpressionStarter<Categoria> predicateCategoria,
        ExpressionStarter<Magazzino> predicateMagazzino,
        ExpressionStarter<PrezzoVendita> predicatePrezzoVendita)
    {
        using (var database = DatabaseContext.NuovoDatabase())
        {
            var query = database.Articoli.AsExpandable();

            if (predicateArticolo != null)
            {
                query = query.Where(predicateArticolo);
            }

            if (predicateFornitore != null)
            {
                query = query.AsExpandable().SelectMany(a => a.ElencoArticoliFornitore.Where(predicateFornitore)).Select(fornitore => fornitore.Articolo);
            }

            if (predicateMagazzino != null)
            {
                query = query.AsExpandable()
                    .SelectMany(articolo => articolo.ElencoMagazzino.Where(predicateMagazzino))
                    .Select(magazzino => magazzino.Articolo);
            }

            if (predicatePrezzoVendita != null)
            {
                query = query.AsExpandable().SelectMany(articolo =>
                    articolo.ElencoPrezzoVendita.Where(predicatePrezzoVendita).Select(vendita => vendita.Articolo));
            }

            if (predicateCategoria != null)
            {
                query = query.AsExpandable()
                    .SelectMany(articolo => articolo.ElencoCategorie.Where(predicateCategoria))
                    .Select(categoria => categoria.Articolo);
            }

            return await query.ToListAsync();
        }
    }

Създавам предиката по този начин

        private ExpressionStarter<ArticoloFornitore> PredicateFornitore()
    {
        var ritornaNull = true;
        var predicate = PredicateBuilder.New<ArticoloFornitore>();
        if (IsEnabledFiltroFornitore && FornitoreSelezionato != null)
        {
            ritornaNull = false;
            predicate = predicate.And(fornitore => fornitore.IdFornitore == FornitoreSelezionato.IdFornitore);
        }
        return ritornaNull ? null : predicate;
    }

    private ExpressionStarter<Categoria> PredicateCategoria()
    {
        var ritornaNull = true;
        var predicate = PredicateBuilder.New<Categoria>();
        if (IsEnabledCategoriaLivello1 && CategoriaLivello1Selezionata != null)
        {
            ritornaNull = false;
            predicate = predicate.And(categoria =>
                categoria.IdCategoriaLv1 == CategoriaLivello1Selezionata.IdCategoriaLv1);
        }

        if (IsEnabledCategoriaLivello2 && CategoriaLivello2Selezionata != null)
        {
            ritornaNull = false;
            predicate = predicate.And(categoria =>
                categoria.IdCategoriaLv2 == CategoriaLivello2Selezionata.IdCategoriaLv2);
        }

        if (IsEnabledCategoriaLivello3 && CategoriaLivello3Selezionata != null)
        {
            ritornaNull = false;
            predicate = predicate.And(categoria =>
                categoria.IdCategoriaLv3 == CategoriaLivello3Selezionata.IdCategoriaLv3);
        }

        if (IsEnabledCategoriaLivello4 && CategoriaLivello4Selezionata != null)
        {
            ritornaNull = false;
            predicate = predicate.And(categoria =>
                categoria.IdCategoriaLv4 == CategoriaLivello4Selezionata.IdCategoriaLv4);
        }

        if (IsEnabledCategoriaLivello5 && CategoriaLivello5Selezionata != null)
        {
            ritornaNull = false;
            predicate = predicate.And(categoria =>
                categoria.IdCategoriaLv5 == CategoriaLivello5Selezionata.IdCategoriaLv5);
        }
        return ritornaNull ? null : predicate;
    }

Опитах този метод на LINQPAD с фиксирани данни вместо PredicateBuilder и заявката работи. Но с PredicateBuilder получавам грешка 1025 на доставчика на данни на .NET Framework.

Как мога да реша? Бих искал да мога да създам динамична заявка, която взема параметрите от интерфейса и връща резултатите.

Надявам се, че можете да ми помогнете.


person Antonio Pisarra    schedule 16.03.2018    source източник
comment
Забравихте да включите код, който използва PredicateBuilder.   -  person Evk    schedule 16.03.2018
comment
Промених публикацията и добавих кода за създаване на предиката   -  person Antonio Pisarra    schedule 16.03.2018
comment
Вижте stackoverflow.com/questions/46706255/. Предлагам да използвате Expression<Func<T, bool>> вместо ExpressionStarter<T>   -  person Ivan Stoev    schedule 16.03.2018


Отговори (1)


EntityFramework изисква вашите предикати да бъдат Expression<Func<T, bool>> вместо Func<T, bool>. Това прави IQueryable.ToExpandable(). Обаче вашите обекти Entity като articolo.ElencoPrezzoVendita са виртуални ICollection обекти, така че предикатът е намален до Func<T, bool>, което е несъвместимо с EF.

Не става точно от езика, но можете да опитате нещо подобно (не знаех структурата на вашите обекти) с база данни, за да сте сигурни, че Expression се използва.

if (predicatePrezzoVendita != null)
{
    query = query.AsExpandable()
        .SelectMany(articolo =>
            database.ElencoPrezzoVendita
                .Where(x => articolo.ForeignKeyID == x.ID)
                .AsExpandable()
                .Where(predicatePrezzoVendita)
                .Select(vendita => vendita.Articolo));
}
person Jono    schedule 16.03.2018
comment
Благодаря много, реших проблема си с този подход. - person Antonio Pisarra; 16.03.2018