Uno dei task più frequenti per uno sviluppatore backend è quello di implementare un metodo di ricerca di una qualsiasi entità su una base dati utilizzando un set di filtri opzionali. Tipicamente. si finisce per craere un metodo che prende in input un oggetto contenente tali filtri, che ha una forma simile a questa:
public class SearchCriteria { public string? FirstName { get; set; } public string? LastName { get; set; } public string? Email { get; set; } public bool? Enabled { get; set; } ... }
Il metodo che effettua la ricerca di utenti su una tabella del database utilizzando un DbContext di EF Core sarà simile a questo:
async Task<IEnumerable<UserEntity>> SearchUsers(SearchCriteria searchCriteria) { var query = _tmsContext.Users; if(!string.IsNullOrWhiteSpace(searchCriteria.FirstName)) query = query.Where(x => x.FirstName.Contains(searchCriteria.FirstName!)); if(!string.IsNullOrWhiteSpace(searchCriteria.LastName)) query = query.Where(x => x. LastName.Contains(searchCriteria. LastName!)); if(!string.IsNullOrWhiteSpace(searchCriteria.Email)) query = query.Where(x => x. Email.Contains(searchCriteria. Email!)); if(searchCriteria.Enabled.HasValue) query = query.Where(x => x.Enabled == searchCriteria.Enabled); return await query.ToListAsync(); }
Per noi dev, less is more e questo vale soprattutto per la quantità di codice da scrivere. Per scrivere meno codice, possiamo creare un extension method che prende in input la condizione in base alla quale applicare il filtro e il filtro. Nel metodo verifichiamo che la condizione sia a true e in caso positivo applichiamo il filtro, in caso negativo non facciamo nulla.
public static class LinqExtensions { public static IQueryable<TSource> WhereIf<TSource>(this IQueryable<TSource> source, bool condition, Expression<Func<TSource, bool>> predicate) => condition ? source.Where(predicate) : source; }
A questo punto, il metodo di ricerca può essere trasformato come segue.
async Task<IEnumerable<UserEntity>> SearchUsers(SearchCriteria searchCriteria) => _tmsContext.Users .WhereIf(!string.IsNullOrWhiteSpace(searchCriteria.FirstName), x => x.FirstName.Contains(searchCriteria.FirstName!)) .WhereIf(!string.IsNullOrWhiteSpace(searchCriteria.LastName), x => x.LastName.Contains(searchCriteria.LastName!)) .WhereIf(!string.IsNullOrWhiteSpace(searchCriteria.Email), x => x.Email.Contains(searchCriteria.Email!)) .WhereIf(searchCriteria.Enabled.HasValue, x => x.Enabled == searchCriteria.Enabled) .ToListAsync();
Il codice del metodo di ricerca che utilizza il nostro extension method è estremamente più compatto e, a seconda dei propri gusti, anche più leggibile.
Commenti
Per inserire un commento, devi avere un account.
Fai il login e torna a questa pagina, oppure registrati alla nostra community.
Approfondimenti
Organizzare il codice JavaScript utilizzando i moduli
Montare blob e file share su Azure App Service
Chiamare un endpoint ASP.NET Core protetto da Certificate Authentication
Utilizzare ChatGPT con Azure OpenAI
Blazor PWA e Offline-First
Utilizzare i nuovi piani dedicati di Azure Container Apps
Sfruttare i tag nell'output cache di ASP.NET Core 7
Utilizzare il metodo reduce in JavaScript
Effettuare update massivi con Entity Framework Core 7
Collegare servizi a Azure Container App con i service connector
Utilizzare la parola chiave file nel codice C#
Gestione degli environment per il deploy con un workflow di GitHub
I più letti di oggi
- devConf 2022 - Online
- .NET Conference Italia 2022 - Milano e Online
- .NET Conference Italia 2021 - Online
- Novità di ASP.NET Core 1.1
- L'object model di Microsoft SharePoint - Seconda parte
- Visual Studio 2019 Live - Milano
- Visual Studio 2010 per l'architetto
- Visual Studio 2017 e il supporto a Docker per ASP.NET Core
- Anteprima di ASP.NET Core 3
- Orchard CMS: dove cresce ASP.NET MVC - Prima parte