Il modo più comune di ricostruire una entity in una action di POST in ASP.NET Core MVC è quello di indicarla come parametro del metodo e lasciare al Model Binder il compito di popolarla:
public async Task<IActionResult> Edit(string id, Person person) { // .. altro codice qui .. }
Questo approccio, tuttavia, ha il pesante limite di creare ogni volta una nuova istanza di Person, che poi - se usiamo Entity Framework, per esempio - dobbiamo in qualche modo riagganciare al Context e marcare come modificata:
_context.Update(person); await _context.SaveChangesAsync();
Questo codice funziona benissimo con entità semplici, ma se l'oggetto Person è più complesso, e per esempio non tutte le proprietà vengono esposte sulla pagina che stiamo creando, rischiamo che queste ultime vengano tutte resettate al loro valore di default.
La ragione, lo ripetiamo, è molto semplice: Person viene ricreata dal model binder da zero, e pertanto conterrà solo le proprietà che effettivamente sono pervenute come parte della request in POST.
Un'alternativa molto valida, invece, è quella di utilizzare TryUpdateModelAsync: a differenza del precendente caso, infatti, questo metodo accetta come parametro un oggetto esistente, che magari abbiamo preventivamente recuperato dal database. In buona sostanza, possiamo scrivere la nostra action di POST come segue:
[HttpPost] [ValidateAntiForgeryToken] public async Task<IActionResult> Edit(int id, Person person) { if (id != person.Id) { return NotFound(); } var original = await _context.Person.SingleAsync(x => x.Id == id); if (ModelState.IsValid && await this.TryUpdateModelAsync(original)) { await _context.SaveChangesAsync(); return RedirectToAction("Index"); } return View(person); }
Il codice in alto, dopo aver recuperato la entity originale dal database (incluse le proprietà non direttamente esposte in pagina), invoca TryUpdateModelAsync passando la entity stessa come parametro. Questo metodo aggiornerà le proprietà provenienti dalla request, lasciando inalterate le altre.
Successivamente, siccome la entity è già collegata al DbContext grazie al metodo SingleAsync, non dobbiamo far altro che invocare SaveChangesAsync per salvare le modifiche.
TryUpdateModelAsync supporta anche il whitelist delle proprietà, analogamente all'attributo Bind, utilizzando la sintassi basata su lambda expression seguente:
this.TryUpdateModelAsync(original, string.Empty, x => x.Name, x => x.Email)
Commenti
Per inserire un commento, devi avere un account.
Fai il login e torna a questa pagina, oppure registrati alla nostra community.
Approfondimenti
Determinare lo stato di un pod in Kubernetes
Usare il versioning con i controller di ASP.NET Core Web API
Eseguire operazioni sui blob con Azure Storage Actions
Utilizzare un service principal per accedere a Azure Container Registry
Load test di ASP.NET Core con k6
Evitare (o ridurre) il repo-jacking sulle GitHub Actions
Sfruttare lo streaming di una chiamata Http da Blazor
Applicare il versioning ai nostri endpoint ASP.NET Core Minimal API
Creazione di componenti personalizzati in React.js con Tailwind CSS
Definire stili a livello di libreria in Angular
Eseguire una GroupBy per entity in Entity Framework
I più letti di oggi
- Cambiare la chiave di partizionamento di Azure Cosmos DB
- ASP.NET 2.0 per tutti
- Definire stili a livello di libreria in Angular
- Utilizzare il trigger SQL con le Azure Function
- Ottimizzare le performance delle collection con le classi FrozenSet e FrozenDictionary
- .NET Conference Italia 2023 - Milano e Online
- Come EF 8 ha ottimizzato le query che usano il metodo Contains
- Ottimizzazione dei block template in Angular 17