Entity Framework - Concurrence

Tout développeur d'accès aux données rencontre des difficultés lorsqu'il répond à la question concernant l'accès concurrentiel des données: "Que se passe-t-il si plusieurs personnes modifient les mêmes données en même temps?"

  • Les plus chanceux d'entre nous traitent avec des règles commerciales qui disent «pas de problème, le dernier gagne».

  • Dans ce cas, la concurrence n'est pas un problème. Plus probablement, ce n'est pas aussi simple que cela, et il n'y a pas de solution miracle pour résoudre tous les scénarios à la fois.

  • Par défaut, Entity Framework prendra le chemin du «dernier des victoires», ce qui signifie que la dernière mise à jour est appliquée même si quelqu'un d'autre a mis à jour les données entre le moment où les données ont été récupérées et l'heure où les données ont été enregistrées.

Prenons un exemple pour mieux le comprendre. L'exemple suivant ajoute une nouvelle colonne VersionNo dans la table Course.

Allez dans le concepteur et faites un clic droit sur la fenêtre du concepteur et sélectionnez le modèle de mise à jour de la base de données…

Vous verrez qu'une autre colonne est ajoutée dans Entité de cours.

Cliquez avec le bouton droit de la souris sur la colonne nouvellement créée VersionNo et sélectionnez Propriétés et changez le ConcurrencyMode en Fixed comme indiqué dans l'image suivante.

Avec le ConcurrencyMode de Course.VersionNo défini sur Fixed, à chaque fois qu'un cours est mis à jour, la commande Update recherche le cours à l'aide de sa EntityKey et de sa propriété VersionNo.

Jetons un coup d'œil à un scénario simple. Deux utilisateurs récupèrent le même cours en même temps et l'utilisateur 1 change le titre de ce cours en Maths et enregistre les modifications avant l'utilisateur 2. Plus tard, lorsque l'utilisateur 2 change le titre de ce cours qui a été récupéré avant que l'utilisateur 1 enregistre ses modifications, en ce sens cas l'utilisateur 2 obtiendra une exception de concurrence"User2: Optimistic Concurrency exception occured".

using System;
using System.Data.Entity;
using System.Data.Entity.Infrastructure;
using System.Linq;

namespace DatabaseFirstDemo {

   class Program {

      static void Main(string[] args) {

         Course c1 = null;
         Course c2 = null;

         //User 1 gets Course

         using (var context = new UniContextEntities()) {
            context.Configuration.ProxyCreationEnabled = false;
            c1 = context.Courses.Where(s ⇒ s.CourseID == 1).Single();
         }

         //User 2 also get the same Course

         using (var context = new UniContextEntities()) {
            context.Configuration.ProxyCreationEnabled = false;
            c2 = context.Courses.Where(s ⇒ s.CourseID == 1).Single();
         }

         //User 1 updates Course Title
         c1.Title = "Edited from user1";

         //User 2 updates Course Title
         c2.Title = "Edited from user2";

         //User 1 saves changes first

         using (var context = new UniContextEntities()) {

            try {
               context.Entry(c1).State = EntityState.Modified;
               context.SaveChanges();
            } catch (DbUpdateConcurrencyException ex) {
               Console.WriteLine("User1: Optimistic Concurrency exception occurred");
            }
         }

         //User 2 saves changes after User 1.
         //User 2 will get concurrency exection
         //because CreateOrModifiedDate is different in the database

         using (var context = new UniContextEntities()) {

            try {
               context.Entry(c2).State = EntityState.Modified;
               context.SaveChanges();
            } catch (DbUpdateConcurrencyException ex) {
               Console.WriteLine("User2: Optimistic Concurrency exception occurred");
            }
         }
      }
   }
}