Entity Framework - Guide rapide

Qu'est-ce que Entity Framework?

Entity Framework a été lancé pour la première fois en 2008, le principal moyen d'interaction de Microsoft entre les applications .NET et les bases de données relationnelles. Entity Framework est un Object Relational Mapper (ORM) qui est un type d'outil qui simplifie le mappage entre les objets de votre logiciel et les tables et colonnes d'une base de données relationnelle.

  • Entity Framework (EF) est un framework ORM open source pour ADO.NET qui fait partie de .NET Framework.

  • Un ORM se charge de créer des connexions de base de données et d'exécuter des commandes, ainsi que de prendre les résultats de la requête et de matérialiser automatiquement ces résultats en tant qu'objets d'application.

  • Un ORM permet également de garder une trace des modifications apportées à ces objets et, lorsque vous y êtes invité, il conservera également ces modifications dans la base de données pour vous.

Pourquoi Entity Framework?

Entity Framework est un ORM et les ORM visent à augmenter la productivité du développeur en réduisant la tâche redondante de persistance des données utilisées dans les applications.

  • Entity Framework peut générer les commandes de base de données nécessaires pour lire ou écrire des données dans la base de données et les exécuter pour vous.

  • Si vous interrogez, vous pouvez exprimer vos requêtes par rapport aux objets de votre domaine à l'aide de LINQ to entités.

  • Entity Framework exécutera la requête appropriée dans la base de données, puis matérialisera les résultats dans des instances de vos objets de domaine pour que vous puissiez travailler dans votre application.

Il existe d'autres ORM sur le marché tels que NHibernate et LLBLGen Pro. La plupart des ORM mappent généralement les types de domaine directement sur le schéma de base de données.

Entity Framework dispose d'une couche de mappage plus granulaire afin que vous puissiez personnaliser les mappages, par exemple, en mappant l'entité unique sur plusieurs tables de base de données ou même plusieurs entités sur une seule table.

  • Entity Framework est la technologie d'accès aux données recommandée par Microsoft pour les nouvelles applications.

  • ADO.NET semble se référer directement à la technologie des ensembles de données et des tableaux de données.

  • Entity Framework est l'endroit où tous les investissements progressifs sont effectués, ce qui est le cas depuis plusieurs années déjà.

  • Microsoft vous recommande d'utiliser Entity Framework sur ADO.NET ou LINQ to SQL pour tout nouveau développement.

Modèle conceptuel

Pour les développeurs habitués au développement axé sur les bases de données, le plus grand changement avec Entity Framework est qu'il vous permet de vous concentrer sur votre domaine d'activité. Que voulez-vous que votre application fasse sans être limitée par ce que la base de données est capable de faire?

  • Avec Entity Framework, le point focal est appelé modèle conceptuel. Il s'agit d'un modèle des objets de votre application, et non d'un modèle de la base de données que vous utilisez pour conserver les données de votre application.

  • Votre modèle conceptuel peut s'aligner sur le schéma de votre base de données ou être très différent.

  • Vous pouvez utiliser un concepteur visuel pour définir votre modèle conceptuel, qui peut ensuite générer les classes que vous utiliserez finalement dans votre application.

  • Vous pouvez simplement définir vos classes et utiliser une fonctionnalité d'Entity Framework appelée Code First. Et puis Entity Framework comprendra le modèle conceptuel.

Dans tous les cas, Entity Framework explique comment passer de votre modèle conceptuel à votre base de données. Ainsi, vous pouvez interroger vos objets de modèle conceptuel et travailler directement avec eux.

traits

Voici les fonctionnalités de base d'Entity Framework. Cette liste est créée en fonction des fonctionnalités les plus notables et également à partir des questions fréquemment posées sur Entity Framework.

  • Entity Framework est un outil Microsoft.
  • Entity Framework est développé en tant que produit Open Source.
  • Entity Framework n'est plus lié ou dépendant du cycle de publication .NET.
  • Fonctionne avec n'importe quelle base de données relationnelle avec un fournisseur Entity Framework valide.
  • Génération de commandes SQL de LINQ vers les entités.
  • Entity Framework créera des requêtes paramétrées.
  • Suit les modifications des objets en mémoire.
  • Permet d'insérer, de mettre à jour et de supprimer la génération de commandes.
  • Fonctionne avec un modèle visuel ou avec vos propres classes.
  • Entity Framework a stocké la prise en charge des procédures.

L'architecture d'Entity Framework, de bas en haut, se compose des éléments suivants:

Fournisseurs de données

Il s'agit de fournisseurs spécifiques à la source, qui résument les interfaces ADO.NET pour se connecter à la base de données lors de la programmation par rapport au schéma conceptuel.

Il traduit les langages SQL courants tels que LINQ via l'arborescence de commandes en expression SQL native et l'exécute sur le système de SGBD spécifique.

Client d'entité

Cette couche expose la couche d'entité à la couche supérieure. Le client d'entité permet aux développeurs de travailler sur des entités sous la forme de lignes et de colonnes à l'aide de requêtes SQL d'entité sans avoir besoin de générer des classes pour représenter le schéma conceptuel. Entity Client affiche les couches de structure d'entité, qui sont la fonctionnalité principale. Ces couches sont appelées modèle de données d'entité.

  • le Storage Layer contient le schéma de base de données complet au format XML.

  • le Entity Layer qui est également un fichier XML définit les entités et les relations.

  • le Mapping layer est un fichier XML qui mappe les entités et les relations définies au niveau de la couche conceptuelle avec les relations et les tables réelles définies au niveau de la couche logique.

  • le Metadata services qui est également représenté dans Entity Client fournit une API centralisée pour accéder aux métadonnées stockées aux couches Entity, Mapping et Storage.

Service d'objets

La couche Object Services est le contexte d'objet, qui représente la session d'interaction entre les applications et la source de données.

  • L'utilisation principale du contexte d'objet est d'effectuer différentes opérations telles que l'ajout, la suppression d'instances d'entités et la sauvegarde de l'état modifié dans la base de données à l'aide de requêtes.

  • Il s'agit de la couche ORM d'Entity Framework, qui représente le résultat des données pour les instances d'objet des entités.

  • Ces services permettent aux développeurs d'utiliser certaines des fonctionnalités riches d'ORM telles que le mappage de clé primaire, le suivi des modifications, etc. en écrivant des requêtes à l'aide de LINQ et d'Entity SQL.

Quoi de neuf dans Entity Framework 6?

Framework possède une API complexe qui vous permet d'avoir un contrôle granulaire sur tout, de sa modélisation à son comportement d'exécution. Une partie d'Entity Framework 5 vit à l'intérieur de .NET. Et une autre partie de celui-ci se trouve dans un assembly supplémentaire distribué à l'aide de NuGet.

  • La fonctionnalité principale d'Entity Framework est intégrée au .NET Framework.

  • La prise en charge de Code First, c'est ce qui permet à Entity Framework d'utiliser des classes au lieu d'un modèle visuel, et une API plus légère pour interagir avec EF se trouve dans le package NuGet.

  • Le noyau est ce qui fournit l'interrogation, le suivi des modifications et toute la transformation de vos requêtes en requêtes SQL ainsi que du retour de données dans les objets.

  • Vous pouvez utiliser le package EF 5 NuGet avec .NET 4 et avec .NET 4.5.

  • Un gros point de confusion - .NET 4.5 a ajouté la prise en charge des énumérations et des données spatiales aux API principales d'Entity Framework, ce qui signifie que si vous utilisez EF 5 avec .NET 4, vous n'obtiendrez pas ces nouvelles fonctionnalités. Vous ne les obtiendrez qu'en combinant EF5 avec .NET 4.5.

Jetons maintenant un œil à Entity Framework 6. Les API de base qui se trouvaient à l'intérieur de .NET dans Entity Framework 6 font désormais partie du package NuGet.

Cela signifie -

  • Tout l'Entity Framework vit à l'intérieur de cet assembly distribué par NuGet

  • Vous ne serez pas dépendant de .NET pour fournir des fonctionnalités spécifiques telles que la prise en charge de l'énumération Entity Framework et la prise en charge des données spéciales.

  • Vous verrez que l'une des fonctionnalités d'EF6 est qu'il prend en charge les énumérations et les données spatiales pour .NET 4

Pour commencer à travailler sur Entity Framework, vous devez installer les outils de développement suivants -

  • Visual Studio 2013 ou supérieur
  • SQL Server 2012 ou supérieur
  • Mises à jour d'Entity Framework à partir du package NuGet

Microsoft fournit une version gratuite de Visual Studio qui contient également SQL Server et peut être téléchargée à partir de www.visualstudio.com .

Installation

Step 1- Une fois le téléchargement terminé, exécutez le programme d'installation. La boîte de dialogue suivante s'affiche.

Step 2 - Cliquez sur le bouton Installer et il lancera le processus d'installation.

Step 3- Une fois le processus d'installation terminé, vous verrez la boîte de dialogue suivante. Fermez cette boîte de dialogue et redémarrez votre ordinateur si nécessaire.

Step 4- Ouvrez Visual Studio à partir du menu Démarrer qui ouvrira la boîte de dialogue suivante. Ce sera un moment pour la première fois pour la préparation.

Step 5 - Une fois que tout est fait, vous verrez la fenêtre principale de Visual studio.

Créons un nouveau projet à partir de Fichier → Nouveau → Projet

Step 1 - Sélectionnez Application console et cliquez sur le bouton OK.

Step 2 - Dans l'Explorateur de solutions, cliquez avec le bouton droit sur votre projet.

Step 3 - Sélectionnez Gérer les packages NuGet comme indiqué dans l'image ci-dessus, ce qui ouvrira la fenêtre suivante dans Visual Studio.

Step 4 - Recherchez Entity Framework et installez la dernière version en appuyant sur le bouton d'installation.

Step 5- Cliquez sur Ok. Une fois l'installation terminée, vous verrez le message suivant dans votre fenêtre de sortie.

Vous êtes maintenant prêt à démarrer votre application.

Dans ce tutoriel, nous utiliserons une simple base de données universitaire. Une base de données universitaire peut être beaucoup plus complexe dans son ensemble, mais à des fins de démonstration et d'apprentissage, nous utilisons la forme la plus simple de cette base de données. Le diagramme suivant contient trois tableaux.

  • Student
  • Course
  • Enrollment

Chaque fois qu'une base de données de termes est utilisée, une chose nous vient directement à l'esprit et c'est un type différent de tables qui a une sorte de relation. Il existe trois types de relations entre les tables et la relation entre les différentes tables dépend de la manière dont les colonnes associées sont définies.

  • Relation un-à-plusieurs
  • Relation plusieurs à plusieurs
  • Relation un à un

Relation un-à-plusieurs

La relation un-à-plusieurs est le type de relation le plus courant. Dans ce type de relation, une ligne de la table A peut avoir plusieurs lignes correspondantes dans la table B, mais une ligne de la table B ne peut avoir qu'une seule ligne correspondante dans la table A. Par exemple, dans le diagramme ci-dessus, la table Student et Enrollment en ont une relation-à-plusieurs, chaque étudiant peut avoir plusieurs inscriptions, mais chaque inscription appartient à un seul étudiant.

Relation plusieurs à plusieurs

Dans une relation plusieurs-à-plusieurs, une ligne de la table A peut avoir plusieurs lignes correspondantes dans la table B, et vice versa. Vous créez une telle relation en définissant une troisième table, appelée table de jonction, dont la clé primaire est constituée des clés étrangères des tables A et B.Par exemple, les tables Student et Course ont une relation plusieurs-à-plusieurs définie par une relation un-à-plusieurs entre chacune de ces tables et la table d'inscription.

Relation un à un

Dans une relation biunivoque, une ligne de la table A ne peut avoir plus d'une ligne correspondante dans la table B, et vice versa. Une relation un-à-un est créée si les deux colonnes associées sont des clés primaires ou ont des contraintes uniques.

Ce type de relation n'est pas courant car la plupart des informations associées de cette manière seraient des tables tout-en-un. Vous pouvez utiliser une relation un-à-un pour -

  • Divisez une table avec plusieurs colonnes.
  • Isolez une partie d'une table pour des raisons de sécurité.
  • Stockez des données de courte durée pouvant être facilement supprimées en supprimant simplement la table.
  • Stockez les informations qui s'appliquent uniquement à un sous-ensemble de la table principale.

Le modèle de données d'entité (EDM) est une version étendue du modèle d'entité-relation qui spécifie le modèle conceptuel des données à l'aide de diverses techniques de modélisation. Il fait également référence à un ensemble de concepts qui décrivent la structure des données, quelle que soit sa forme stockée.

EDM prend en charge un ensemble de types de données primitifs qui définissent des propriétés dans un modèle conceptuel. Nous devons considérer 3 parties principales qui forment la base d'Entity Framework et collectivement, il est connu sous le nom de modèle de données d'entité. Voici les trois parties principales de l'EDM.

  • Le modèle de schéma de stockage
  • Le modèle conceptuel
  • Le modèle de cartographie

Le modèle de schéma de stockage

Le modèle de stockage également appelé couche de définition de schéma de stockage (SSDL) représente la représentation schématique du magasin de données backend.

Le modèle conceptuel

Le modèle conceptuel également appelé couche de définition de schéma conceptuel (CSDL) est le modèle d'entité réel, sur lequel nous écrivons nos requêtes.

Le modèle de cartographie

La couche de mappage n'est qu'un mappage entre le modèle conceptuel et le modèle de stockage.

Le schéma logique et son mappage avec le schéma physique sont représentés comme un EDM.

  • Visual Studio fournit également Entity Designer, pour la création visuelle de l'EDM et de la spécification de mappage.

  • La sortie de l'outil est le fichier XML (* .edmx) spécifiant le schéma et le mappage.

  • Le fichier Edmx contient des artefacts de métadonnées Entity Framework.

Langage de définition de schéma

ADO.NET Entity Framework utilise un langage de définition de données basé sur XML appelé Schema Definition Language (SDL) pour définir le schéma EDM.

  • Le SDL définit les types simples comme les autres types primitifs, notamment String, Int32, Double, Decimal et DateTime.

  • Une énumération, qui définit une carte de valeurs et de noms primitifs, est également considérée comme un type simple.

  • Les énumérations sont prises en charge à partir de la version 5.0 du framework uniquement.

  • Les types complexes sont créés à partir d'une agrégation d'autres types. Une collection de propriétés de ces types définit un type d'entité.

Le modèle de données a principalement trois concepts clés pour décrire la structure des données -

  • Type d'entité
  • Type d'association
  • Property

Type d'entité

Le type d'entité est le bloc de construction fondamental pour décrire la structure des données dans EDM.

  • Dans un modèle conceptuel, les types d'entités sont construits à partir de propriétés et décrivent la structure des concepts de niveau supérieur, tels que les étudiants et les inscriptions dans une application métier.

  • Une entité représente un objet spécifique tel qu'un étudiant ou une inscription spécifique.

  • Chaque entité doit avoir une clé d'entité unique dans un ensemble d'entités. Un ensemble d'entités est une collection d'instances d'un type d'entité spécifique. Les ensembles d'entités (et les ensembles d'associations) sont regroupés logiquement dans un conteneur d'entités.

  • L'héritage est pris en charge avec les types d'entité, c'est-à-dire qu'un type d'entité peut être dérivé d'un autre.

Type d'association

C'est un autre élément fondamental pour décrire les relations dans l'EDM. Dans un modèle conceptuel, une association représente une relation entre deux types d'entités tels que Student et Enrollment.

  • Chaque association a deux extrémités d'association qui spécifient les types d'entités impliqués dans l'association.

  • Chaque extrémité d'association spécifie également une multiplicité de fin d'association qui indique le nombre d'entités qui peuvent se trouver à cette extrémité de l'association.

  • Une multiplicité de fin d'association peut avoir une valeur de un (1), zéro ou un (0..1) ou plusieurs (*).

  • Les entités situées à une extrémité d'une association sont accessibles via les propriétés de navigation ou via des clés étrangères si elles sont exposées sur un type d'entité.

Propriété

Les types d'entité contiennent des propriétés qui définissent leur structure et leurs caractéristiques. Par exemple, un type d'entité étudiant peut avoir des propriétés telles que l'ID étudiant, le nom, etc.

Une propriété peut contenir des données primitives (telles qu'une chaîne, un entier ou une valeur booléenne) ou des données structurées (telles qu'un type complexe).

Entity Framework vous permet d'interroger, d'insérer, de mettre à jour et de supprimer des données à l'aide d'objets CLR (Common Language Runtime) appelés entités. Entity Framework mappe les entités et les relations définies dans votre modèle à une base de données. Il fournit également des installations pour -

  • Matérialiser les données renvoyées par la base de données en tant qu'objets d'entité
  • Suivre les modifications apportées aux objets
  • Gérer la concurrence
  • Propager les modifications d'objets dans la base de données
  • Lier des objets à des contrôles

La classe principale responsable de l'interaction avec les données en tant qu'objets est System.Data.Entity.DbContext. L'API DbContext n'est pas publiée dans le cadre du .NET Framework. Afin d'être plus flexible et plus fréquent avec la publication de nouvelles fonctionnalités dans Code First et l'API DbContext, l'équipe Entity Framework distribue EntityFramework.dll via la fonctionnalité de distribution NuGet de Microsoft.

  • NuGet vous permet d'ajouter des références à vos projets .NET en extrayant les DLL pertinentes directement dans votre projet à partir du Web.

  • Une extension Visual Studio appelée Library Package Manager fournit un moyen simple d'extraire l'assembly approprié du Web dans vos projets.

  • L'API DbContext vise principalement à simplifier votre interaction avec Entity Framework.

  • Cela réduit également le nombre de méthodes et de propriétés dont vous avez besoin pour accéder aux tâches couramment utilisées.

  • Dans les versions précédentes d'Entity Framework, ces tâches étaient souvent compliquées à découvrir et à coder.

  • La classe de contexte gère les objets d'entité pendant l'exécution, ce qui comprend le remplissage des objets avec des données d'une base de données, le suivi des modifications et la persistance des données dans la base de données.

Définition d'une classe dérivée DbContext

La méthode recommandée pour travailler avec le contexte consiste à définir une classe qui dérive de DbContext et expose les propriétés DbSet qui représentent des collections des entités spécifiées dans le contexte. Si vous travaillez avec EF Designer, le contexte sera généré pour vous. Si vous travaillez avec Code First, vous écrirez généralement vous-même le contexte.

Le code suivant est un exemple simple qui montre qu'UniContext est dérivé de DbContext.

  • Vous pouvez utiliser des propriétés automatiques avec DbSet telles que getter et setter.

  • Cela crée également un code beaucoup plus propre, mais vous n'êtes pas obligé de l'utiliser pour créer un DbSet lorsque vous n'avez aucune autre logique à appliquer.

public class UniContext : DbContext {
   public UniContext() : base("UniContext") { }
   public DbSet<Student> Students { get; set; }
   public DbSet<Enrollment> Enrollments { get; set; }
   public DbSet<Course> Courses { get; set; }
}
  • Auparavant, EDM générait des classes de contexte dérivées de la classe ObjectContext.

  • Travailler avec ObjectContext était un peu complexe.

  • DbContext est un wrapper autour d'ObjectContext qui est en fait similaire à ObjectContext et est utile et facile dans tous les modèles de développement tels que Code First, Model First et Database First.

Requêtes

Il existe trois types de requêtes que vous pouvez utiliser, telles que -

  • Ajout d'une nouvelle entité.
  • Modification ou mise à jour des valeurs de propriété d'une entité existante.
  • Supprimer une entité existante.

Ajout de nouvelles entités

L'ajout d'un nouvel objet avec Entity Framework est aussi simple que la construction d'une nouvelle instance de votre objet et son enregistrement à l'aide de la méthode Add sur DbSet. Le code suivant est destiné lorsque vous souhaitez ajouter un nouvel étudiant à la base de données.

private static void AddStudent() {

   using (var context = new UniContext()) {

      var student = new Student {
         LastName = "Khan", 
         FirstMidName = "Ali", 
         EnrollmentDate = DateTime.Parse("2005-09-01") 
      };

      context.Students.Add(student); 
      context.SaveChanges();

   }
}

Modification des entités existantes

La modification des objets existants est aussi simple que la mise à jour de la valeur attribuée aux propriétés que vous souhaitez modifier et l'appel de SaveChanges. Dans le code suivant, le nom de famille d'Ali a été changé de Khan à Aslam.

private static void AddStudent() {

   private static void ChangeStudent() {

      using (var context = new UniContext()) {

         var student = (from d in context.Students
            where d.FirstMidName == "Ali" select d).Single();
         student.LastName = "Aslam";
         context.SaveChanges();

      }
   }
}

Suppression d'entités existantes

Pour supprimer une entité à l'aide d'Entity Framework, vous utilisez la méthode Remove sur DbSet. Supprimez les œuvres des entités existantes et nouvellement ajoutées. L'appel de Remove sur une entité qui a été ajoutée mais pas encore enregistrée dans la base de données annulera l'ajout de l'entité. L'entité est supprimée du suivi des modifications et n'est plus suivie par le DbContext. L'appel de Remove sur une entité existante qui fait l'objet d'un suivi des modifications enregistrera l'entité pour suppression la prochaine fois que SaveChanges sera appelé. L'exemple suivant montre une instance dans laquelle l'étudiant est supprimé de la base de données dont le prénom est Ali.

private static void DeleteStudent() {

   using (var context = new UniContext()) {
      var bay = (from d in context.Students where d.FirstMidName == "Ali" select d).Single();
      context.Students.Remove(bay);
      context.SaveChanges();
   }
}

Dans Entity Framework, il existe deux types d'entités qui permettent aux développeurs d'utiliser leurs propres classes de données personnalisées avec le modèle de données sans apporter de modifications aux classes de données elles-mêmes.

  • Entités POCO
  • Proxy dynamique

Entités POCO

  • POCO signifie objets CLR «simples» qui peuvent être utilisés comme objets de domaine existants avec votre modèle de données.

  • Les classes de données POCO qui sont mappées à des entités sont définies dans un modèle de données.

  • Il prend également en charge la plupart des mêmes comportements de requête, d'insertion, de mise à jour et de suppression que les types d'entité générés par les outils de modèle de données d'entité.

  • Vous pouvez utiliser le modèle POCO pour générer des types d'entités ignorant la persistance à partir d'un modèle conceptuel.

Jetons un coup d'œil à l'exemple suivant de modèle de données d'entité conceptuelle.

Pour générer des entités POCO pour le modèle d'entité ci-dessus -

Step 1- Faites un clic droit sur la fenêtre du concepteur. Il affichera la boîte de dialogue suivante.

Step 2 - Sélectionnez l'élément Ajouter une génération de code ...

Step 3 - Sélectionnez le générateur EF 6.x DbContext, écrivez le nom, puis cliquez sur le bouton Ajouter.

Vous verrez dans votre explorateur de solutions que les modèles POCODemo.Context.tt et POCODemo.tt sont générés.

Le POCODemo.Context génère le DbContext et les ensembles d'objets que vous pouvez retourner et utiliser pour interroger, par exemple pour le contexte, les étudiants et les cours, etc.

L'autre modèle traite de tous les types Étudiant, Cours, etc. Voici le code de la classe Étudiant qui est généré automatiquement à partir du modèle d'entité.

namespace ConsoleApplication1 {

   using System;
   using System.Collections.Generic;

   public partial class Student {

      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         "CA2214:DoNotCallOverridableMethodsInConstructors")]

      public Student() {
         this.Enrollments = new HashSet<Enrollment>();
      }

      public int ID { get; set; }
      public string LastName { get; set; }
      public string FirstMidName { get; set; }
      public System.DateTime EnrollmentDate { get; set; }

      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         CA2227:CollectionPropertiesShouldBeReadOnly")]

      public virtual ICollection<Enrollment> Enrollments { get; set; }

   }
}

Des classes similaires sont générées pour les tables de cours et d'inscription à partir du modèle d'entité.

Proxy dynamique

Lors de la création d'instances de types d'entité POCO, Entity Framework crée souvent des instances d'un type dérivé généré dynamiquement qui agit comme un proxy pour l'entité. On peut également dire qu'il s'agit d'une classe proxy d'exécution comme une classe wrapper d'entité POCO.

  • Vous pouvez remplacer certaines propriétés de l'entité pour effectuer automatiquement des actions lors de l'accès à la propriété.

  • Ce mécanisme est utilisé pour prendre en charge le chargement différé des relations et le suivi automatique des modifications.

  • Cette technique s'applique également aux modèles créés avec Code First et EF Designer.

Si vous souhaitez que Entity Framework prenne en charge le chargement différé des objets associés et effectue le suivi des modifications dans les classes POCO, les classes POCO doivent répondre aux exigences suivantes:

  • La classe de données personnalisée doit être déclarée avec un accès public.

  • La classe de données personnalisée ne doit pas être scellée.

  • La classe de données personnalisée ne doit pas être abstraite.

  • La classe de données personnalisée doit avoir un constructeur public ou protégé qui n'a pas de paramètres.

  • Utilisez un constructeur protégé sans paramètres si vous souhaitez que la méthode CreateObject soit utilisée pour créer un proxy pour l'entité POCO.

  • L'appel de la méthode CreateObject ne garantit pas la création du proxy: la classe POCO doit respecter les autres exigences décrites dans cette rubrique.

  • La classe ne peut pas implémenter les interfaces IEntityWithChangeTracker ou IEntityWithRelationships car les classes proxy implémentent ces interfaces.

  • L'option ProxyCreationEnabled doit être définie sur true.

L'exemple suivant est celui d'une classe d'entité proxy dynamique.

public partial class Course {

   public Course() {
      this.Enrollments = new HashSet<Enrollment>();
   }

   public int CourseID { get; set; }
   public string Title { get; set; }
   public int Credits { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Pour désactiver la création d'objets proxy, définissez la valeur de la propriété ProxyCreationEnabled sur false.

Dans les bases de données relationnelles, la relation est une situation qui existe entre des tables de base de données relationnelles via des clés étrangères. Une clé étrangère (FK) est une colonne ou une combinaison de colonnes utilisée pour établir et appliquer un lien entre les données de deux tables. Le diagramme suivant contient trois tableaux.

  • Student
  • Course
  • Enrollment

Dans le diagramme ci-dessus, vous pouvez voir une sorte d'association / relation entre les tables. Il existe trois types de relations entre les tables et la relation entre les différentes tables dépend de la manière dont les colonnes associées sont définies.

  • Relation un-à-plusieurs
  • Relation plusieurs à plusieurs
  • Relation un à un

Relation un-à-plusieurs

  • Une relation un-à-plusieurs est le type de relation le plus courant.

  • Dans ce type de relation, une ligne de la table A peut avoir plusieurs lignes correspondantes dans la table B, mais une ligne de la table B ne peut avoir qu'une seule ligne correspondante dans la table A.

  • La clé étrangère est définie dans la table qui représente la fin plusieurs de la relation.

  • Par exemple, dans le diagramme ci-dessus, les tables Étudiant et Inscription ont une relation un à plusieurs, chaque étudiant peut avoir plusieurs inscriptions, mais chaque inscription appartient à un seul étudiant.

Dans le cadre d'entité, ces relations peuvent également être créées avec du code. Voici un exemple de classes d'étudiants et d'inscription qui sont associées à une relation un à plusieurs.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

public class Enrollment {

   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
	
   public Grade? Grade { get; set; }
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }
}

Dans le code ci-dessus, vous pouvez voir que la classe Student contient la collection d'Enrollment, mais que la classe Enrollment a un seul objet Student.

Relation plusieurs à plusieurs

Dans une relation plusieurs-à-plusieurs, une ligne de la table A peut avoir plusieurs lignes correspondantes dans la table B, et vice versa.

  • Vous pouvez créer une telle relation en définissant une troisième table, appelée table de jonction, dont la clé primaire est constituée des clés étrangères de la table A et de la table B.

  • Par exemple, les tables Student et Course ont une relation plusieurs-à-plusieurs qui est définie par une relation un-à-plusieurs entre chacune de ces tables et la table d'inscription.

Le code suivant contient la classe Course et les deux classes ci-dessus, c'est-à-dire Student et Enrollment.

public class Course {
   [DatabaseGenerated(DatabaseGeneratedOption.None)]
	
   public int CourseID { get; set; }
   public string Title { get; set; }
	
   public int Credits { get; set; } 
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez voir que la classe de cours et la classe Student ont des collections d'objets d'inscription qui établissent une relation plusieurs-à-plusieurs via l'inscription de classe de jonction.

Relation un à un

  • Dans une relation biunivoque, une ligne de la table A ne peut avoir plus d'une ligne correspondante dans la table B, et vice versa.

  • Une relation un-à-un est créée si les deux colonnes associées sont des clés primaires ou ont des contraintes uniques.

  • Dans une relation un-à-un, la clé primaire agit en outre comme une clé étrangère et il n'y a pas de colonne de clé étrangère distincte pour l'une ou l'autre table.

Ce type de relation n'est pas courant car la plupart des informations liées de cette manière se trouveraient toutes dans un seul tableau. Vous pouvez utiliser une relation un-à-un pour -

  • Divisez une table avec plusieurs colonnes.
  • Isolez une partie d'une table pour des raisons de sécurité.
  • Stockez des données de courte durée pouvant être facilement supprimées en supprimant simplement la table.
  • Stockez les informations qui s'appliquent uniquement à un sous-ensemble de la table principale.

Le code suivant consiste à ajouter un autre nom de classe StudentProfile qui contient l'ID et le mot de passe de messagerie de l'étudiant.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
   public virtual StudentProfile StudentProfile { get; set; }
}

public class StudentProfile {

   public StudentProfile() {}
   public int ID { get; set; }
   public string Email { get; set; }
   public string Password { get; set; }
	
   public virtual Student Student { get; set; }
}

Vous pouvez voir que la classe d'entité Student contient la propriété de navigation StudentProfile et StudentProfile contient la propriété de navigation Student.

Chaque étudiant n'a qu'un seul e-mail et mot de passe pour se connecter au domaine universitaire. Ces informations peuvent être ajoutées à la table Student, mais pour des raisons de sécurité, elles sont séparées dans une autre table.

Durée de vie

La durée de vie d'un contexte commence lorsque l'instance est créée et se termine lorsque l'instance est supprimée ou récupérée.

  • La durée de vie du contexte est une décision très cruciale à prendre lorsque nous utilisons des ORM.

  • Le contexte fonctionne comme un cache d'entités, ce qui signifie qu'il contient des références à toutes les entités chargées qui peuvent augmenter très rapidement la consommation de mémoire et qu'il peut également provoquer des fuites de mémoire.

  • Dans le diagramme ci-dessous, vous pouvez voir le niveau supérieur du flux de travail de données d'une application à une base de données via le contexte et vice versa.

Cycle de vie de l'entité

Le cycle de vie de l'entité décrit le processus dans lequel une entité est créée, ajoutée, modifiée, supprimée, etc. Les entités ont de nombreux états au cours de sa durée de vie. Avant de voir comment récupérer l'état de l'entité, examinons ce qu'est l'état de l'entité. L'état est une énumération de typeSystem.Data.EntityState qui déclare les valeurs suivantes -

  • Added: L'entité est marquée comme ajoutée.

  • Deleted: L'entité est marquée comme supprimée.

  • Modified: L'entité a été modifiée.

  • Unchanged: L'entité n'a pas été modifiée.

  • Detached: L'entité n'est pas suivie.

Changements d'état dans le cycle de vie de l'entité

Parfois, l'état des entités est défini automatiquement par le contexte, mais il peut également être modifié manuellement par le développeur. Même si toutes les combinaisons de commutations d'un état à un autre sont possibles, certaines d'entre elles n'ont pas de sens. Par exemple,Added entité à la Deleted état, ou vice versa.

Discutons des différents états.

État inchangé

  • Lorsqu'une entité est inchangée, elle est liée au contexte mais elle n'a pas été modifiée.

  • Par défaut, une entité extraite de la base de données est dans cet état.

  • Lorsqu'une entité est attachée au contexte (avec la méthode Attach), elle est de la même manière dans l'état Inchangé.

  • Le contexte ne peut pas suivre les modifications apportées aux objets auxquels il ne fait pas référence, donc lorsqu'ils sont attachés, il suppose qu'ils sont inchangés.

État détaché

  • Détaché est l'état par défaut d'une entité nouvellement créée car le contexte ne peut pas suivre la création d'un objet dans votre code.

  • Cela est vrai même si vous instanciez l'entité à l'intérieur d'un bloc using du contexte.

  • Détaché est même l'état des entités récupérées de la base de données lorsque le suivi est désactivé.

  • Lorsqu'une entité est détachée, elle n'est pas liée au contexte, son état n'est donc pas suivi.

  • Il peut être éliminé, modifié, utilisé en combinaison avec d'autres classes ou utilisé de toute autre manière dont vous pourriez avoir besoin.

  • Comme il n'y a pas de suivi de contexte, cela n'a aucune signification pour Entity Framework.

État ajouté

  • Lorsqu'une entité est dans l'état Ajouté, vous avez peu d'options. En fait, vous ne pouvez que le détacher du contexte.

  • Naturellement, même si vous modifiez une propriété, l'état reste Ajouté, car le déplacer vers Modifié, Inchangé ou Supprimé n'a aucun sens.

  • Il s'agit d'une nouvelle entité et n'a aucune correspondance avec une ligne de la base de données.

  • C'est une condition préalable fondamentale pour être dans l'un de ces états (mais cette règle n'est pas appliquée par le contexte).

État modifié

  • Lorsqu'une entité est modifiée, cela signifie qu'elle était à l'état inchangé et qu'une propriété a été modifiée.

  • Une fois qu'une entité entre dans l'état Modifié, elle peut passer à l'état Détaché ou Supprimé, mais elle ne peut pas revenir à l'état Inchangé même si vous restaurez manuellement les valeurs d'origine.

  • Il ne peut même pas être changé en Ajouté, sauf si vous détachez et ajoutez l'entité au contexte, car une ligne avec cet ID existe déjà dans la base de données et vous obtiendriez une exception d'exécution lors de sa conservation.

État supprimé

  • Une entité entre dans l'état Supprimé car elle était inchangée ou modifiée, puis la méthode DeleteObject a été utilisée.

  • C'est l'état le plus restrictif, car il est inutile de passer de cet état à une autre valeur que Détaché.

le usingsi vous souhaitez que toutes les ressources contrôlées par le contexte soient supprimées à la fin du bloc. Lorsque vous utilisez leusing , puis le compilateur crée automatiquement un bloc try / finally et appelle dispose dans le bloc finally.

using (var context = new UniContext()) {

   var student = new Student {
      LastName = "Khan", 
      FirstMidName = "Ali", 
      EnrollmentDate = DateTime.Parse("2005-09-01")
   };

   context.Students.Add(student);
   context.SaveChanges();
}

Lorsque vous travaillez avec un contexte de longue durée, tenez compte des éléments suivants:

  • Au fur et à mesure que vous chargez plus d'objets et leurs références en mémoire, la consommation de mémoire du contexte peut augmenter rapidement. Cela peut entraîner des problèmes de performances.

  • N'oubliez pas de disposer du contexte lorsqu'il n'est plus nécessaire.

  • Si une exception fait que le contexte est dans un état irrécupérable, toute l'application peut se terminer.

  • Les chances de rencontrer des problèmes liés à la concurrence augmentent à mesure que l'écart entre le moment où les données sont interrogées et mises à jour augmente.

  • Lorsque vous travaillez avec des applications Web, utilisez une instance de contexte par requête.

  • Lorsque vous travaillez avec Windows Presentation Foundation (WPF) ou Windows Forms, utilisez une instance de contexte par formulaire. Cela vous permet d'utiliser la fonctionnalité de suivi des modifications fournie par le contexte.

Règles de base

Web Applications

  • C'est maintenant une pratique courante et recommandée que pour les applications Web, le contexte soit utilisé par requête.

  • Dans les applications web, nous traitons des requêtes qui sont très courtes mais qui contiennent toutes les transactions du serveur, elles ont donc la durée appropriée pour le contexte à vivre.

Desktop Applications

  • Pour les applications de bureau, comme Win Forms / WPF, etc., le contexte est utilisé par formulaire / boîte de dialogue / page.

  • Puisque nous ne voulons pas avoir le contexte en tant que singleton pour notre application, nous le supprimerons lorsque nous passerons d'un formulaire à un autre.

  • De cette façon, nous gagnerons beaucoup de capacités du contexte et ne souffrirons pas des implications de contextes de longue durée.

L'Entity Framework propose trois approches pour créer un modèle d'entité et chacune a ses propres avantages et inconvénients.

  • Code d'abord
  • La base de données d'abord
  • Modèle d'abord

Dans ce chapitre, nous décrirons brièvement la première approche du code. Certains développeurs préfèrent travailler avec le Designer dans Code tandis que d'autres préfèrent simplement travailler avec leur code. Pour ces développeurs, Entity Framework dispose d'un workflow de modélisation appelé Code First.

  • Le workflow de modélisation Code First cible une base de données qui n'existe pas et Code First la créera.

  • Il peut également être utilisé si vous avez une base de données vide et que Code First ajoutera également de nouvelles tables.

  • Code First vous permet de définir votre modèle à l'aide des classes C # ou VB.Net.

  • Une configuration supplémentaire peut éventuellement être effectuée à l'aide d'attributs sur vos classes et propriétés ou à l'aide d'une API fluide.

Pourquoi Code First?

  • Code First est vraiment composé d'un ensemble de pièces de puzzle. Premièrement, vos classes de domaine.

  • Les classes de domaine n'ont rien à voir avec Entity Framework. Ce ne sont que les éléments de votre domaine professionnel.

  • Entity Framework a donc un contexte qui gère l'interaction entre ces classes et votre base de données.

  • Le contexte n'est pas spécifique à Code First. C'est une fonctionnalité Entity Framework.

  • Code First ajoute un générateur de modèle qui inspecte vos classes gérées par le contexte, puis utilise un ensemble de règles ou de conventions pour déterminer comment ces classes et les relations décrivent un modèle et comment ce modèle doit être mappé à votre base de données.

  • Tout cela se produit au moment de l'exécution. Vous ne verrez jamais ce modèle, c'est juste en mémoire.

  • Code First a la possibilité d'utiliser ce modèle pour créer une base de données si nécessaire.

  • Il peut également mettre à jour la base de données si le modèle change, à l'aide d'une fonctionnalité appelée Migrations Code First.

Dans ce chapitre, apprenons à créer un modèle de données d'entité dans le concepteur à l'aide du flux de travail appelé Model First.

  • Model First est idéal lorsque vous démarrez un nouveau projet où la base de données n'existe même pas encore.

  • Le modèle est stocké dans un fichier EDMX et peut être affiché et modifié dans Entity Framework Designer.

  • Dans Model First, vous définissez votre modèle dans un concepteur Entity Framework, puis générez du SQL, qui créera un schéma de base de données correspondant à votre modèle, puis vous exécuterez le SQL pour créer le schéma dans votre base de données.

  • Les classes avec lesquelles vous interagissez dans votre application sont automatiquement générées à partir du fichier EDMX.

Voici un exemple simple de création d'un nouveau projet de console à l'aide de l'approche Model First.

Step 1 - Ouvrez Visual Studio et sélectionnez Fichier → Nouveau → Projet

Step 2 - Sélectionnez Installé → Modèles → Visual C # → Windows dans le volet gauche, puis dans le volet central, sélectionnez Application console.

Step 3 - Entrez EFModelFirstDemo dans le champ Nom.

Step 4 - Pour créer un modèle, faites d'abord un clic droit sur votre projet de console dans l'explorateur de solutions et sélectionnez Ajouter → Nouveaux éléments…

La boîte de dialogue suivante s'ouvre.

Step 5 - Sélectionnez ADO.NET Entity Data Model dans le volet central et entrez le nom ModelFirstDemoDB dans le champ Nom.

Step 6 - Cliquez sur le bouton Ajouter qui lancera la boîte de dialogue Assistant de modèle de données d'entité.

Step 7- Sélectionnez le modèle Empty EF Designer et cliquez sur le bouton Suivant. Le concepteur Entity Framework s'ouvre avec un modèle vide. Nous pouvons maintenant commencer à ajouter des entités, des propriétés et des associations au modèle.

Step 8- Cliquez avec le bouton droit sur l'aire de conception et sélectionnez Propriétés. Dans la fenêtre Propriétés, remplacez le nom du conteneur d'entité par ModelFirstDemoDBContext.

Step 9 - Cliquez avec le bouton droit de la souris sur l'aire de conception et sélectionnez Ajouter nouveau → Entité…

La boîte de dialogue Ajouter une entité s'ouvre comme indiqué dans l'image suivante.

Step 10 - Entrez Student comme nom d'entité et Student Id comme nom de propriété et cliquez sur OK.

Step 11 - Cliquez avec le bouton droit sur la nouvelle entité sur l'aire de conception et sélectionnez Ajouter nouveau → Propriété scalaire, entrez Nom comme nom de la propriété.

Step 12 - Entrez FirstName, puis ajoutez deux autres propriétés scalaires telles que LastName et EnrollmentDate.

Step 13 - Ajoutez deux autres cours d'entités et inscription en suivant toutes les étapes mentionnées ci-dessus et ajoutez également des propriétés scalaires comme indiqué dans les étapes suivantes.

Step 14 - Nous avons trois entités dans Visual Designer, ajoutons une association ou une relation entre elles.

Step 15 - Cliquez avec le bouton droit de la souris sur l'aire de conception et sélectionnez Ajouter nouveau → Association…

Step 16 - Faites une extrémité de la relation pointer vers l'étudiant avec une multiplicité de l'un et l'autre extrémité vers l'inscription avec une multiplicité de plusieurs.

Step 17 - Cela signifie qu'un étudiant a de nombreuses inscriptions et que l'inscription appartient à un seul étudiant.

Step 18 - Assurez-vous que la case Ajouter les propriétés de la clé étrangère à l'entité «Post» est cochée et cliquez sur OK.

Step 19 - De même, ajoutez une autre association entre le cours et l'inscription.

Step 20 - Votre modèle de données ressemblera à l'écran suivant après l'ajout d'associations entre les entités.

Nous avons maintenant un modèle simple à partir duquel nous pouvons générer une base de données et l'utiliser pour lire et écrire des données. Allons-y et générons la base de données.

Step 1 - Cliquez avec le bouton droit de la souris sur l'aire de conception et sélectionnez Générer la base de données à partir du modèle…

Step 2 - Vous pouvez sélectionner une base de données existante ou créer une nouvelle connexion en cliquant sur Nouvelle connexion…

Step 3 - Pour créer une nouvelle base de données, cliquez sur Nouvelle connexion…

Step 4 - Entrez le nom du serveur et le nom de la base de données.

Step 5 - Cliquez sur Suivant.

Step 6- Cliquez sur Terminer. Cela ajoutera le fichier * .edmx.sql dans le projet. Vous pouvez exécuter des scripts DDL dans Visual Studio en ouvrant le fichier .sql, puis cliquez avec le bouton droit et sélectionnez Exécuter.

Step 7 - La boîte de dialogue suivante s'affiche pour vous connecter à la base de données.

Step 8 - En cas d'exécution réussie, vous verrez le message suivant.

Step 9 - Allez dans l'explorateur de serveur, vous verrez que la base de données est créée avec trois tables qui sont spécifiées.

Ensuite, nous devons permuter notre modèle pour générer du code qui utilise l'API DbContext.

Step 1 - Cliquez avec le bouton droit de la souris sur un emplacement vide de votre modèle dans EF Designer et sélectionnez Ajouter un élément de génération de code…

Vous verrez que la boîte de dialogue Ajouter un nouvel élément suivante s'ouvre.

Step 2 - Sélectionnez EF 6.x DbContext Generator dans le volet central et entrez ModelFirstDemoModel dans le champ Nom.

Step 3 - Vous verrez dans votre explorateur de solutions que les modèles ModelFirstDemoModel.Context.tt et ModelFirstDemoModel.tt sont générés.

Le ModelFirstDemoModel.Context génère le DbCcontext et les jeux d'objets que vous pouvez retourner et utiliser pour interroger, par exemple pour le contexte, les étudiants et les cours, etc.

L'autre modèle traite de tous les types d'étudiant, de cours, etc. Voici la classe d'étudiant, qui est générée automatiquement à partir du modèle d'entité.

Voici le code C # dans lequel certaines données sont entrées et extraites de la base de données.

using System;
using System.Linq;

namespace EFModelFirstDemo {

   class Program {

      static void Main(string[] args) {

         using (var db = new ModelFirstDemoDBContext()) {

            // Create and save a new Student

            Console.Write("Enter a name for a new Student: ");
            var firstName = Console.ReadLine();

            var student = new Student {
               StudentID = 1,
               FirstName = firstName
            };
				
            db.Students.Add(student);
            db.SaveChanges();
				
            var query = from b in db.Students
               orderby b.FirstName select b;

            Console.WriteLine("All student in the database:");

            foreach (var item in query) {
               Console.WriteLine(item.FirstName);
            }

            Console.WriteLine("Press any key to exit...");
            Console.ReadKey();
         }
      }
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

Enter a name for a new Student:
Ali Khan
All student in the database:
Ali Khan
Press any key to exit...

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans ce chapitre, apprenons à créer un modèle de données d'entité avec l'approche Database First.

  • L'approche Database First fournit une alternative aux approches Code First et Model First du modèle de données d'entité. Il crée des codes de modèle (classes, propriétés, DbContext, etc.) à partir de la base de données du projet et ces classes deviennent le lien entre la base de données et le contrôleur.

  • L'approche Database First crée le cadre d'entité à partir d'une base de données existante. Nous utilisons toutes les autres fonctionnalités, telles que la synchronisation modèle / base de données et la génération de code, de la même manière que nous les avons utilisées dans l'approche Model First.

Prenons un exemple simple. Nous avons déjà une base de données qui contient 3 tables comme le montre l'image suivante.

Step 1 - Créons un nouveau projet de console avec le nom DatabaseFirstDemo.

Step 2 - Pour créer le modèle, faites d'abord un clic droit sur votre projet de console dans l'explorateur de solutions et sélectionnez Ajouter → Nouveaux éléments…

Step 3 - Sélectionnez ADO.NET Entity Data Model dans le volet central et entrez le nom DatabaseFirstModel dans le champ Nom.

Step 4 - Cliquez sur le bouton Ajouter qui lancera la boîte de dialogue Assistant de modèle de données d'entité.

Step 5 - Sélectionnez EF Designer dans la base de données et cliquez sur le bouton Suivant.

Step 6 - Sélectionnez la base de données existante et cliquez sur Suivant.

Step 7 - Choisissez Entity Framework 6.x et cliquez sur Suivant.

Step 8 - Sélectionnez toutes les vues de tables et la procédure stockée que vous souhaitez inclure et cliquez sur Terminer.

Vous verrez que le modèle d'entité et les classes POCO sont générés à partir de la base de données.

Récupérons maintenant tous les étudiants de la base de données en écrivant le code suivant dans le fichier program.cs.

using System;
using System.Linq;

namespace DatabaseFirstDemo {

   class Program {

      static void Main(string[] args) {

         using (var db = new UniContextEntities()) {

            var query = from b in db.Students
               orderby b.FirstMidName select b;

            Console.WriteLine("All All student in the database:");

            foreach (var item in query) {
               Console.WriteLine(item.FirstMidName +" "+ item.LastName);
            }

            Console.WriteLine("Press any key to exit...");
            Console.ReadKey();
         }
      }
   }
}

Lorsque le programme ci-dessus est exécuté, vous recevrez la sortie suivante -

All student in the database:
Ali Khan
Arturo   finand
Bill Gates
Carson Alexander
Gytis Barzdukas
Laura Norman
Meredith Alonso
Nino Olivetto
Peggy Justice
Yan Li
Press any key to exit...

Lorsque le programme ci-dessus est exécuté, vous verrez tous les noms des étudiants qui ont été précédemment saisis dans la base de données.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans ce chapitre, concentrons-nous sur la création de modèles avec Designer ou Database First ou simplement en utilisant Code First. Voici quelques conseils qui vous aideront à décider quel workflow de modélisation choisir.

  • Nous avons déjà vu des exemples de modélisation Code First, de modélisation Database First et d'un workflow de modélisation Model First.

  • Les flux de travail Database First et Model First utilisaient le Designer, mais l'un commence par la base de données pour créer un modèle et l'autre commence au modèle pour créer une base de données.

  • Pour les développeurs qui ne souhaitent pas utiliser Visual Designer plus la génération de code, Entity Framework a un flux de travail complètement différent appelé Code First.

  • Le flux de travail typique de Code First est idéal pour les toutes nouvelles applications où vous n'avez même pas de base de données. Vous définissez vos classes et votre code, puis laissez Code First déterminer à quoi devrait ressembler votre base de données.

  • Il est également possible de démarrer Code First avec une base de données et cela rend Code First un peu contradictoire. Mais il existe un outil pour vous permettre de procéder à l'ingénierie inverse d'une base de données en classes, ce qui est un excellent moyen d'avoir une longueur d'avance sur le codage.

Compte tenu de ces options, regardons l'arbre de décision.

  • Si vous préférez travailler avec un concepteur visuel dans du code généré, vous souhaiterez choisir l'un des flux de travail impliquant EF Designer. Si votre base de données existe déjà, Database First est votre chemin.

  • Si vous souhaitez utiliser un concepteur visuel sur un tout nouveau projet sans base de données, vous souhaiterez utiliser Model First.

  • Si vous souhaitez simplement travailler avec du code et non avec un concepteur, alors Code First est probablement pour vous, avec la possibilité d'utiliser l'outil qui rétroconce la base de données en classes.

  • Si vous avez des classes existantes, le mieux est de les utiliser avec Code First.

Dans les chapitres précédents, vous avez appris trois manières différentes de définir un modèle de données d'entité.

  • Deux d'entre eux, Database First et Model First, dépendaient du concepteur Entity Framework combiné à la génération de code.

  • Le troisième, Code First, vous permet d'ignorer un concepteur visuel et d'écrire simplement votre propre code.

  • Quel que soit le chemin que vous choisissez, vous vous retrouverez avec des classes de domaine et une ou plusieurs classes Entity Framework DbContext vous permettent de récupérer et de conserver les données pertinentes pour ces classes.

L'API DbContext de vos applications est utilisée comme un pont entre vos classes et votre base de données. Le DbContext est l'une des classes les plus importantes de Entity Framework.

  • Il permet d'exprimer et d'exécuter des requêtes.

  • Il prend les résultats de la requête de la base de données et les transforme en instances de nos classes de modèle.

  • Il peut suivre les modifications des entités, y compris l'ajout et la suppression, puis déclencher la création d'instructions d'insertion, de mise à jour et de suppression qui sont envoyées à la base de données à la demande.

Voici les classes de contexte d'annonce de domaine sur lesquelles nous allons effectuer différentes opérations dans ce chapitre. C'est le même exemple que nous avons créé dans le chapitre, Database First Approach.

Implémentation de classe de contexte

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

namespace DatabaseFirstDemo {

   public partial class UniContextEntities : DbContext {

      public UniContextEntities(): base("name = UniContextEntities") {}

      protected override void OnModelCreating(DbModelBuilder modelBuilder) {
         throw new UnintentionalCodeFirstException();
      }

      public virtual DbSet<Course> Courses { get; set; }
      public virtual DbSet<Enrollment> Enrollments { get; set; }
      public virtual DbSet<Student> Students { get; set; }
   }
}

Implémentation des classes de domaine

Classe de cours

namespace DatabaseFirstDemo {

   using System;
   using System.Collections.Generic;
	
   public partial class Course {

      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         "CA2214:DoNotCallOverridableMethodsInConstructors")]

      public Course() {
         this.Enrollments = new HashSet<Enrollment>();
      }
	
      public int CourseID { get; set; }
      public string Title { get; set; }
      public int Credits { get; set; }
	
      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         "CA2227:CollectionPropertiesShouldBeReadOnly")]
			
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }
}

Classe étudiante

namespace DatabaseFirstDemo {

   using System;
   using System.Collections.Generic; 

   public partial class Student {

      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         "CA2214:DoNotCallOverridableMethodsInConstructors")]

      public Student() {
         this.Enrollments = new HashSet<Enrollment>();
      }

      public int ID { get; set; }
      public string LastName { get; set; }
      public string FirstMidName { get; set; }
      public System.DateTime EnrollmentDate { get; set; }

      [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", 
         "CA2227:CollectionPropertiesShouldBeReadOnly")]
			
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }
}

Classe d'inscription

namespace DatabaseFirstDemo {

   using System;
   using System.Collections.Generic; 

   public partial class Enrollment {

      public int EnrollmentID { get; set; }
      public int CourseID { get; set; }
      public int StudentID { get; set; }
      public Nullable<int> Grade { get; set; }
		
      public virtual Course Course { get; set; }
      public virtual Student Student { get; set; }
   }
}

Créer une opération

L'ajout d'un nouvel objet avec Entity Framework est aussi simple que la construction d'une nouvelle instance de votre objet et son enregistrement à l'aide de la méthode Add sur DbSet. Le code suivant vous permet d'ajouter un nouvel étudiant à la base de données.

class Program {

   static void Main(string[] args) {

      var newStudent = new Student();

      //set student name

      newStudent.FirstMidName = "Bill";
      newStudent.LastName = "Gates";
      newStudent.EnrollmentDate = DateTime.Parse("2015-10-21");
      newStudent.ID = 100;

      //create DBContext object

      using (var dbCtx = new UniContextEntities()) {

         //Add Student object into Students DBset
         dbCtx.Students.Add(newStudent);

         // call SaveChanges method to save student into database
         dbCtx.SaveChanges();
      }
   }
}

Opération de mise à jour

La modification des objets existants est aussi simple que la mise à jour de la valeur attribuée aux propriétés que vous souhaitez modifier et l'appel de SaveChanges. Par exemple, le code suivant est utilisé pour changer le nom de famille d'Ali de Khan à Aslam.

using (var context = new UniContextEntities()) {

   var student = (from d in context.Students where d.FirstMidName == "Ali" select d).Single();
   student.LastName = "Aslam";
   context.SaveChanges();
}

Supprimer l'opération

Pour supprimer une entité à l'aide d'Entity Framework, vous utilisez la méthode Remove sur DbSet. Supprimez les œuvres des entités existantes et nouvellement ajoutées. L'appel de Remove sur une entité qui a été ajoutée mais pas encore enregistrée dans la base de données annulera l'ajout de l'entité. L'entité est supprimée du suivi des modifications et n'est plus suivie par le DbContext. L'appel de Remove sur une entité existante qui fait l'objet d'un suivi des modifications enregistrera l'entité pour suppression la prochaine fois que SaveChanges sera appelé. L'exemple suivant est un code dans lequel l'étudiant est supprimé de la base de données dont le prénom est Ali.

using (var context = new UniContextEntities()) {
   var bay = (from d in context.Students where d.FirstMidName == "Ali" select d).Single();
   context.Students.Remove(bay);
   context.SaveChanges();
}

Lire l'opération

La lecture des données existantes de la base de données est très simple. Voici le code dans lequel toutes les données de la table des étudiants sont récupérées, puis un programme sera affiché avec le prénom et le nom des étudiants dans l'ordre alphabétique.

using (var db = new UniContextEntities()) {

   var query = from b in db.Students orderby b.FirstMidName select b;
   Console.WriteLine("All All student in the database:");

   foreach (var item in query) {
      Console.WriteLine(item.FirstMidName +" "+ item.LastName);
   }

   Console.WriteLine("Press any key to exit...");
   Console.ReadKey();
}

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 Course Entity.

Cliquez avec le bouton droit de la souris sur la colonne nouvellement créée VersionNo et sélectionnez Propriétés et modifiez 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");
            }
         }
      }
   }
}

Dans toutes les versions d'Entity Framework, chaque fois que vous exécutez SaveChanges()pour insérer, mettre à jour ou supprimer la base de données, le framework encapsulera cette opération dans une transaction. Lorsque vous appelez SaveChanges, le contexte démarre automatiquement une transaction et la valide ou la restaure selon que la persistance a réussi.

  • Tout cela est transparent pour vous et vous n'aurez jamais besoin de vous en occuper.

  • Cette transaction ne dure que le temps d'exécuter l'opération, puis se termine.

  • Lorsque vous exécutez une autre opération de ce type, une nouvelle transaction démarre.

Entity Framework 6 fournit les éléments suivants:

Database.BeginTransaction ()

  • C'est une méthode simple et plus facile dans un DbContext existant pour démarrer et terminer des transactions pour les utilisateurs.

  • Il permet à plusieurs opérations d'être combinées dans la même transaction et donc soit toutes sont validées, soit toutes sont annulées en une seule.

  • Il permet également à l'utilisateur de spécifier plus facilement le niveau d'isolement de la transaction.

Database.UseTransaction ()

  • Il permet au DbContext d'utiliser une transaction qui a été démarrée en dehors de Entity Framework.

Jetons un coup d'œil à l'exemple suivant où plusieurs opérations sont effectuées dans une seule transaction. Le code est comme -

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         using (var dbContextTransaction = context.Database.BeginTransaction()) {

            try {

               Student student = new Student() {
                  ID = 200, 
                  FirstMidName = "Ali", 
                  LastName = "Khan", 
                  EnrollmentDate = DateTime.Parse("2015-12-1")
               };

               context.Students.Add(student);

               context.Database.ExecuteSqlCommand(@"UPDATE Course SET Title = 
                  'Calculus'" + "WHERE CourseID = 1045");

               var query = context.Courses.Where(c ⇒ c.CourseID == 1045);

               foreach (var item in query) {
                  Console.WriteLine(item.CourseID.ToString()
                     + " " + item.Title + " " + item.Credits);
               }

               context.SaveChanges();
               var query1 = context.Students.Where(s ⇒ s.ID == 200);

               foreach (var item in query1) {
                  Console.WriteLine(item.ID.ToString() 
                     + " " + item.FirstMidName + " " + item.LastName);
               }

               dbContextTransaction.Commit();
            } catch (Exception) {
               dbContextTransaction.Rollback();
            }

         }
      }
   }
}
  • Le début d'une transaction nécessite que la connexion au magasin sous-jacente soit ouverte.

  • Donc, appeler Database.BeginTransaction () ouvrira la connexion, si elle n'est pas déjà ouverte.

  • Si DbContextTransaction a ouvert la connexion, il la fermera lorsque Dispose () est appelé.

Une vue est un objet qui contient des données obtenues par une requête prédéfinie. Une vue est un objet virtuel ou une table dont le jeu de résultats est dérivé d'une requête. Il est très similaire à une vraie table car il contient des colonnes et des lignes de données. Voici quelques utilisations typiques des vues -

  • Filtrer les données des tables sous-jacentes
  • Filtrer les données à des fins de sécurité
  • Centralisez les données réparties sur plusieurs serveurs
  • Créer un ensemble de données réutilisable

Les vues peuvent être utilisées de la même manière que vous pouvez utiliser des tables. Pour utiliser la vue en tant qu'entité, vous devez d'abord ajouter des vues de base de données à EDM. Après avoir ajouté des vues à votre modèle, vous pouvez l'utiliser de la même manière que les entités normales, à l'exception des opérations de création, de mise à jour et de suppression.

Jetons un coup d'œil, comment ajouter des vues dans le modèle à partir de la base de données.

Step 1 - Créez un nouveau projet d'application console.

Step 2 - Cliquez avec le bouton droit sur le projet dans l'explorateur de solutions et sélectionnez Ajouter → Nouvel élément.

Step 3 - Sélectionnez ADO.NET Entity Data Model dans le volet central et entrez le nom ViewModel dans le champ Nom.

Step 4 - Cliquez sur le bouton Ajouter qui lancera la boîte de dialogue Assistant de modèle de données d'entité.

Step 5 - Sélectionnez EF Designer dans la base de données et cliquez sur le bouton Suivant.

Step 6 - Sélectionnez la base de données existante et cliquez sur Suivant.

Step 7 - Choisissez Entity Framework 6.x et cliquez sur Suivant.

Step 8 - Sélectionnez les tables et les vues de votre base de données et cliquez sur Terminer.

Vous pouvez voir dans la fenêtre du concepteur qu'une vue est créée et vous pouvez l'utiliser dans le programme en tant qu'entité.

Dans l'explorateur de solutions, vous pouvez voir que la classe MyView est également générée à partir de la base de données.

Prenons un exemple dans lequel toutes les données sont extraites de la vue. Voici le code -

class Program {

   static void Main(string[] args) {

      using (var db = new UniContextEntities()) {

         var query = from b in db.MyViews
            orderby b.FirstMidName select b;

         Console.WriteLine("All student in the database:");

         foreach (var item in query) {
            Console.WriteLine(item.FirstMidName + " " + item.LastName);
         }

         Console.WriteLine("Press any key to exit...");
         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

All student in the database:
Ali Khan
Arturo   finand
Bill Gates
Carson Alexander
Gytis Barzdukas
Laura Norman
Meredith Alonso
Nino Olivetto
Peggy Justice
Yan Li
Press any key to exit...

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Un index est une structure de données sur disque basée sur des tables et des vues. Les index rendent la récupération des données plus rapide et plus efficace, dans la plupart des cas. Cependant, la surcharge d'une table ou d'une vue avec des index peut affecter de manière désagréable les performances d'autres opérations telles que les insertions ou les mises à jour.

  • L'indexation est la nouvelle fonctionnalité de la structure d'entité où vous pouvez améliorer les performances de votre application Code First en réduisant le temps nécessaire pour interroger les données de la base de données.

  • Vous pouvez ajouter des index à votre base de données en utilisant le Index attribut et remplacer la valeur par défaut Unique et Clustered paramètres pour obtenir l'index le mieux adapté à votre scénario.

Jetons un coup d'œil au code suivant dans lequel l'attribut Index est ajouté dans la classe Course pour CourseID.

public partial class Course {

   public Course() {
      this.Enrollments = new HashSet<Enrollment>();
   }

   [Index]
   public int CourseID { get; set; }
   public string Title { get; set; }
	
   public int Credits { get; set; }
   public byte[] VersionNo { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

La clé créée ci-dessus est non unique, non groupée. Il existe des surcharges disponibles pour remplacer ces valeurs par défaut -

  • Pour faire d'un index un index clusterisé, vous devez spécifier IsClustered = true

  • De même, vous pouvez également faire d'un index un index unique en spécifiant IsUnique = true

Jetons un coup d'œil au code C # suivant où un index est clusterisé et unique.

public partial class Course {

   public Course() {
      this.Enrollments = new HashSet<Enrollment>();
   }

   [Index(IsClustered = true, IsUnique = true)]
   public int CourseID { get; set; }
   public string Title { get; set; }
	
   public int Credits { get; set; }
   public byte[] VersionNo { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

L'attribut index peut être utilisé pour créer un index unique dans la base de données. Cependant, cela ne signifie pas qu'EF sera en mesure de raisonner sur l'unicité de la colonne lorsqu'il s'agit de relations, etc. Cette fonctionnalité est généralement appelée prise en charge des «contraintes uniques».

Entity Framework vous permet d'utiliser des procédures stockées dans le modèle de données d'entité au lieu ou en combinaison avec sa génération de commande automatique.

  • Vous pouvez utiliser des procédures stockées pour exécuter une logique prédéfinie sur les tables de base de données, et de nombreuses organisations ont mis en place des stratégies qui nécessitent l'utilisation de ces procédures stockées.

  • Il peut également spécifier qu'EF doit utiliser vos procédures stockées pour insérer, mettre à jour ou supprimer des entités.

  • Bien que les commandes générées dynamiquement soient sécurisées, efficaces et généralement aussi bonnes ou meilleures que celles que vous pouvez écrire vous-même, il existe de nombreux cas où des procédures stockées existent déjà et les pratiques de votre entreprise peuvent restreindre l'utilisation directe des tables.

  • Vous pouvez également vouloir simplement avoir un contrôle explicite sur ce qui est exécuté sur le magasin et préférer créer des procédures stockées.

L'exemple suivant crée un nouveau projet à partir de Fichier → Nouveau → Projet.

Step 1 - Sélectionnez l'application console dans le volet central et saisissez StoredProceduresDemo dans le champ de nom.

Step 2 - Dans l'explorateur de serveur, cliquez avec le bouton droit sur votre base de données.

Step 3 - Sélectionnez Nouvelle requête et entrez le code suivant dans l'éditeur T-SQL pour ajouter une nouvelle table dans votre base de données.

IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = 
   OBJECT_ID(N'[dbo].[StudentGrade]') AND type in (N'U'))

BEGIN

   CREATE TABLE [dbo].[StudentGrade](

      [EnrollmentID] [int] IDENTITY(1,1) NOT NULL,
      [CourseID] [int] NOT NULL,
      [StudentID] [int] NOT NULL,
      [Grade] [decimal](3, 2) NULL,

      CONSTRAINT [PK_StudentGrade] PRIMARY KEY CLUSTERED (
         [EnrollmentID] ASC
      )

      WITH (IGNORE_DUP_KEY = OFF) ON [PRIMARY]

   ) ON [PRIMARY]

END
GO

Step 4 - Cliquez avec le bouton droit de la souris sur l'éditeur et sélectionnez Exécuter.

Step 5- Faites un clic droit sur votre base de données et cliquez sur Actualiser. Vous verrez la table nouvellement ajoutée dans votre base de données.

Step 6 - Dans l'explorateur de serveur, cliquez à nouveau avec le bouton droit sur votre base de données.

Step 7 - Sélectionnez Nouvelle requête et entrez le code suivant dans l'éditeur T-SQL pour ajouter une procédure stockée dans votre base de données, qui renverra les notes des étudiants.

IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id = 
   OBJECT_ID(N'[dbo].[GetStudentGrades]') AND type in (N'P', N'PC'))

BEGIN

   EXEC dbo.sp_executesql @statement = N'
   CREATE PROCEDURE [dbo].[GetStudentGrades]
   @StudentID int
   AS
   SELECT EnrollmentID, Grade, CourseID, StudentID FROM dbo.StudentGrade 
   WHERE StudentID = @StudentID
   '
END
GO

Step 8 - Cliquez avec le bouton droit de la souris sur l'éditeur et sélectionnez Exécuter.

Step 9- Faites un clic droit sur votre base de données et cliquez sur Actualiser. Vous verrez qu'une procédure stockée est créée dans votre base de données.

Step 10 - Cliquez avec le bouton droit sur le nom du projet dans l'Explorateur de solutions et sélectionnez Ajouter → Nouvel élément.

Step 11 - Sélectionnez ensuite ADO.NET Entity Data Model dans le volet Modèles.

Step 12 - Entrez SPModel comme nom, puis cliquez sur Ajouter.

Step 13 - Dans la boîte de dialogue Choisir le contenu du modèle, sélectionnez Concepteur EF à partir de la base de données, puis cliquez sur Suivant.

Step 14 - Sélectionnez votre base de données et cliquez sur Suivant.

Step 15 - Dans la boîte de dialogue Choisissez vos objets de base de données, cliquez sur les tables, les vues.

Step 16 - Sélectionnez la fonction GetStudentGradesForCourse située sous le nœud Procédures et fonctions stockées et cliquez sur Terminer.

Step 17 - Sélectionnez Affichage → Autres fenêtres → Navigateur de modèles de données d'entité et cliquez avec le bouton droit sur GetStudentGrades sous Importations de fonctions et sélectionnez Modifier.

Il produira le dialogue suivant.

Step 18 - Cliquez sur le bouton radio Entités et sélectionnez StudentGrade dans la liste déroulante comme type de retour de cette procédure stockée et cliquez sur OK.

Jetons un coup d'œil au code C # suivant dans lequel toutes les notes seront récupérées en transmettant l'ID étudiant en tant que paramètre dans la procédure stockée GetStudentGrades.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         int studentID = 22;
         var studentGrades = context.GetStudentGrades(studentID);

         foreach (var student in studentGrades) {
            Console.WriteLine("Course ID: {0}, Title: {1}, Grade: {2} ", 
               student.CourseID, student.Course.Title, student.Grade);
         }

         Console.ReadKey();

      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante -

Course ID: 4022, Title: Microeconomics, Grade: 3.00
Course ID: 4041, Title: Macroeconomics, Grade: 3.50

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans ce chapitre, voyons comment apporter des modifications aux entités qui ne sont pas suivies par un contexte. Les entités qui ne sont pas suivies par un contexte sont appelées entités «déconnectées».

  • Pour la plupart des applications à un seul niveau, où l'interface utilisateur et les couches d'accès à la base de données s'exécutent dans le même processus d'application, vous n'effectuerez probablement que des opérations sur des entités suivies par un contexte.

  • Les opérations sur des entités déconnectées sont beaucoup plus courantes dans les applications N-Tier.

  • Les applications N-Tier impliquent de récupérer certaines données sur un serveur et de les renvoyer, via le réseau, à une machine cliente.

  • L'application cliente manipule ensuite ces données avant de les renvoyer au serveur pour qu'elles soient persistantes.

Voici les deux étapes à suivre avec le graphe d'entités déconnectées ou même une seule entité déconnectée.

  • Attachez des entités à la nouvelle instance de contexte et rendez compte du contexte sur ces entités.

  • Définissez manuellement les EntityStates appropriés sur ces entités.

Jetons un coup d'œil au code suivant dans lequel l'entité Student est ajoutée avec deux entités Enrollment.

class Program {

   static void Main(string[] args) {

      var student = new Student {

         ID = 1001,
         FirstMidName = "Wasim",
         LastName = "Akram", 

         EnrollmentDate = DateTime.Parse("2015-10-10"), 
            Enrollments = new List<Enrollment> {

               new Enrollment{EnrollmentID = 2001,CourseID = 4022, StudentID = 1001 },
               new Enrollment{EnrollmentID = 2002,CourseID = 4025, StudentID = 1001 },
         }
      };

      using (var context = new UniContextEntities()) {

         context.Students.Add(student);
         Console.WriteLine("New Student ({0} {1}): {2}", 
            student.FirstMidName, student.LastName, context.Entry(student).State);

         foreach (var enrollment in student.Enrollments) {
            Console.WriteLine("Enrollment ID: {0} State: {1}", 
               enrollment.EnrollmentID, context.Entry(enrollment).State);
         }

         Console.WriteLine("Press any key to exit...");
         Console.ReadKey();
      }
   } 
}
  • Le code construit une nouvelle instance Student, qui fait également référence à deux nouvelles instances Enrollment dans sa propriété Enrollments.

  • Ensuite, le nouvel étudiant est ajouté à un contexte à l'aide de la méthode Add.

  • Une fois l'étudiant ajouté, le code utilise la méthode DbContext.Entry pour accéder aux informations de suivi des modifications dont Entity Framework dispose sur le nouvel étudiant.

  • À partir de ces informations de suivi des modifications, la propriété State est utilisée pour écrire l'état actuel de l'entité.

  • Ce processus est ensuite répété pour chacune des inscriptions nouvellement créées qui sont référencées par le nouvel étudiant. Si vous exécutez l'application, vous recevrez la sortie suivante -

New Student   (Wasim  Akram): Added
Enrollment ID: 2001 State: Added
Enrollment ID: 2002 State: Added
Press any key to exit...

Alors que DbSet.Add est utilisé pour indiquer à Entity Framework les nouvelles entités, DbSet.Attach est utilisé pour indiquer à Entity Framework les entités existantes. La méthode Attach marquera une entité dans l'état Inchangé.

Jetons un coup d'œil au code C # suivant dans lequel une entité déconnectée est attachée avec DbContext.

class Program {

   static void Main(string[] args) {

      var student = new Student {

         ID = 1001,
         FirstMidName = "Wasim",
         LastName = "Akram",
         EnrollmentDate = DateTime.Parse("2015-10-10"), 

         Enrollments = new List<Enrollment> {
            new Enrollment { EnrollmentID = 2001, CourseID = 4022, StudentID = 1001 },
            new Enrollment { EnrollmentID = 2002, CourseID = 4025, StudentID = 1001 },
         }
			
      };

      using (var context = new UniContextEntities()) {

         context.Students.Attach(student);
         Console.WriteLine("New Student ({0} {1}): {2}", 
            student.FirstMidName, student.LastName, context.Entry(student).State);

         foreach (var enrollment in student.Enrollments) {
            Console.WriteLine("Enrollment ID: {0} State: {1}", enrollment.EnrollmentID, 
               context.Entry(enrollment).State);
         }

         Console.WriteLine("Press any key to exit...");
         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est exécuté avec la méthode Attach (), vous recevrez la sortie suivante.

New Student   (Wasim  Akram): Unchanged
Enrollment ID: 2001 State: Unchanged
Enrollment ID: 2002 State: Unchanged
Press any key to exit...

Dans ce chapitre, apprenons à mapper des fonctions table (TVF) à l'aide d'Entity Framework Designer et à appeler un TVF à partir d'une requête LINQ.

  • Les TVF ne sont actuellement prises en charge que dans le flux de travail Database First.

  • Il a été introduit pour la première fois dans Entity Framework version 5.

  • Pour utiliser les TVF, vous devez cibler .NET Framework 4.5 ou supérieur.

  • Il est très similaire aux procédures stockées mais avec une différence clé, c'est-à-dire que le résultat d'un TVF est composable. Cela signifie que les résultats d'un TVF peuvent être utilisés dans une requête LINQ alors que les résultats d'une procédure stockée ne le peuvent pas.

Jetons un coup d'œil à l'exemple suivant de création d'un nouveau projet à partir de Fichier → Nouveau → Projet.

Step 1 - Sélectionnez l'application console dans le volet central et entrez TableValuedFunctionDemo dans le champ de nom.

Step 2 - Dans l'explorateur de serveur, cliquez avec le bouton droit sur votre base de données.

Step 3 - Sélectionnez Nouvelle requête et entrez le code suivant dans l'éditeur T-SQL pour ajouter une nouvelle table dans votre base de données.

IF NOT EXISTS (SELECT * FROM sys.objects WHERE object_id =
   OBJECT_ID(N'[dbo].[StudentGrade]') AND type in (N'U'))

BEGIN

   CREATE TABLE [dbo].[StudentGrade](

      [EnrollmentID] [int] IDENTITY(1,1) NOT NULL,
      [CourseID] [int] NOT NULL,
      [StudentID] [int] NOT NULL,
      [Grade] [decimal](3, 2) NULL,

      CONSTRAINT [PK_StudentGrade] PRIMARY KEY CLUSTERED ([EnrollmentID] ASC)

      WITH (IGNORE_DUP_KEY = OFF) ON [PRIMARY]

   ) ON [PRIMARY]

END
GO

Step 4 - Cliquez avec le bouton droit de la souris sur l'éditeur et sélectionnez Exécuter.

Step 5- Faites un clic droit sur votre base de données et cliquez sur Actualiser. Vous verrez la table nouvellement ajoutée dans votre base de données.

Step 6- Créez maintenant une fonction qui retournera les notes des étudiants pour le cours. Entrez le code suivant dans l'éditeur T-SQL.

CREATE FUNCTION [dbo].[GetStudentGradesForCourse]

(@CourseID INT)

RETURNS TABLE

RETURN
   SELECT [EnrollmentID],
      [CourseID],
      [StudentID],
      [Grade]
   FROM   [dbo].[StudentGrade]
   WHERE  CourseID = @CourseID

Step 7 - Cliquez avec le bouton droit de la souris sur l'éditeur et sélectionnez Exécuter.

Vous pouvez maintenant voir que la fonction est créée.

Step 8 - Cliquez avec le bouton droit sur le nom du projet dans l'Explorateur de solutions et sélectionnez Ajouter → Nouvel élément.

Step 9 - Sélectionnez ensuite ADO.NET Entity Data Model dans le volet Modèles.

Step 10 - Entrez TVFModel comme nom, puis cliquez sur Ajouter.

Step 11 - Dans la boîte de dialogue Choisir le contenu du modèle, sélectionnez Concepteur EF à partir de la base de données, puis cliquez sur Suivant.

Step 12 - Sélectionnez votre base de données et cliquez sur Suivant.

Step 13 - Dans la boîte de dialogue Choisissez vos objets de base de données, sélectionnez des tables, des vues.

Step 14 - Sélectionnez la fonction GetStudentGradesForCourse située sous le nœud Procédures et fonctions stockées et cliquez sur Terminer.

Step 15 - Sélectionnez Affichage → Autres fenêtres → Navigateur de modèles de données d'entité et cliquez avec le bouton droit sur GetStudentGradesForCourse sous Importations de fonctions et sélectionnez Modifier.

Vous verrez la boîte de dialogue suivante.

Step 16 - Cliquez sur le bouton radio Entités et sélectionnez Inscription dans la liste déroulante comme type de retour de cette fonction et cliquez sur OK.

Jetons un coup d'œil au code C # suivant dans lequel toutes les notes des étudiants inscrits au cours ID = 4022 dans la base de données seront récupérées.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         var CourseID = 4022;

         // Return all the best students in the Microeconomics class.
         var students = context.GetStudentGradesForCourse(CourseID);

         foreach (var result in students) {
            Console.WriteLine("Student ID:  {0}, Grade: {1}",
               result.StudentID, result.Grade);
         }

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante -

Student ID: 1, Grade: 2
Student ID: 4, Grade: 4
Student ID: 9, Grade: 3.5

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans Entity Framework, vous pouvez interroger vos classes d'entité à l'aide de LINQ. Vous pouvez également exécuter des requêtes à l'aide de SQL brut directement sur la base de données à l'aide de DbCOntext. Les techniques peuvent être appliquées de la même manière aux modèles créés avec Code First et EF Designer.

Requête SQL sur une entité existante

La méthode SqlQuery sur DbSet permet d'écrire une requête SQL brute qui retournera des instances d'entité. Les objets retournés seront suivis par le contexte comme ils le seraient s'ils étaient renvoyés par une requête LINQ. Par exemple -

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         var students = context.Students.SqlQuery("SELECT * FROM dbo.Student").ToList();

         foreach (var student in students) {
            string name = student.FirstMidName + " " + student.LastName;
            Console.WriteLine("ID: {0}, Name: {1}, \tEnrollment Date {2} ",
               student.ID, name, student.EnrollmentDate.ToString());
         }

         Console.ReadKey();
      }
   }
}

Le code ci-dessus récupérera tous les étudiants de la base de données.

Requête SQL pour les types non-entité

Une requête SQL renvoyant des instances de n'importe quel type, y compris les types primitifs, peut être créée à l'aide de la méthode SqlQuery sur la classe Database. Par exemple -

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         var studentNames = context.Database.SqlQuery
            <string>("SELECT FirstMidName FROM dbo.Student").ToList();

         foreach (var student in studentNames) {
            Console.WriteLine("Name: {0}", student);
         }

         Console.ReadKey();
      }
   }
}

Commandes SQL vers la base de données

La méthode ExecuteSqlCommnad est utilisée pour envoyer des commandes sans requête à la base de données, telles que la commande Insérer, Mettre à jour ou Supprimer. Jetons un coup d'œil au code suivant dans lequel le prénom de l'élève est mis à jour sous la forme ID = 1

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         //Update command

         int noOfRowUpdated = context.Database.ExecuteSqlCommand("Update 
            student set FirstMidName = 'Ali' where ID = 1");

         context.SaveChanges();

         var student = context.Students.SqlQuery("SELECT * FROM
            dbo.Student where ID = 1").Single();

         string name = student.FirstMidName + " " + student.LastName;

         Console.WriteLine("ID: {0}, Name: {1}, \tEnrollment Date {2} ", 
            student.ID, name, student.EnrollmentDate.ToString());

         Console.ReadKey();
      }
   }
}

Le code ci-dessus récupérera le prénom de tous les étudiants de la base de données.

Dans Entity Framework, cette fonctionnalité vous permettra de définir une propriété sur une classe de domaine qui est un type enum et de la mapper à une colonne de base de données de type entier. Entity Framework convertira ensuite la valeur de la base de données vers et à partir de l'énumération appropriée au fur et à mesure qu'il interroge et enregistre les données.

  • Les types énumérés présentent toutes sortes d'avantages lorsque vous travaillez avec des propriétés qui ont un nombre fixe de réponses.

  • La sécurité et la fiabilité d'une application augmentent toutes deux lorsque vous utilisez des énumérations.

  • L'énumération rend beaucoup plus difficile pour l'utilisateur de faire des erreurs et les problèmes tels que les attaques par injection sont inexistants.

  • Dans Entity Framework, une énumération peut avoir les types sous-jacents suivants -

    • Byte
    • Int16
    • Int32
    • Int64
    • SByte
  • Le type sous-jacent par défaut des éléments d'énumération est int.

  • Par défaut, le premier énumérateur a la valeur 0 et la valeur de chaque énumérateur successif est augmentée de 1.

Jetons un coup d'œil à l'exemple suivant dans lequel nous allons créer une entité dans Designer, puis ajouter des propriétés.

Step 1 - Créer un nouveau projet à partir de l'option de menu Fichier → Nouveau → Projet.

Step 2 - Dans le volet gauche, sélectionnez l'application console.

Step 3 - Entrez EFEnumDemo comme nom du projet et cliquez sur OK.

Step 4 - Cliquez avec le bouton droit sur le nom du projet dans l'Explorateur de solutions et sélectionnez l'option de menu Ajouter → Nouvel élément.

Step 5 - Sélectionnez ADO.NET Entity Data Model dans le volet Modèles.

Step 6 - Entrez EFEnumModel.edmx comme nom de fichier, puis cliquez sur Ajouter.

Step 7 - Sur la page Assistant de modèle de données d'entité, sélectionnez Modèle de concepteur EF vide.

Step 8 - Cliquez sur Terminer

Step 9 - Cliquez ensuite avec le bouton droit sur la fenêtre du concepteur et sélectionnez Ajouter → Entité.

La boîte de dialogue Nouvelle entité apparaît comme illustré dans l'image suivante.

Step 10 - Entrez Department comme nom d'entité et DeptID comme nom de propriété, laissez le type de propriété Int32 et cliquez sur OK.

Step 11 - Faites un clic droit sur l'entité et sélectionnez Ajouter nouveau → Propriété scalaire.

Step 12 - Renommez la nouvelle propriété en DeptName.

Step 13 - Changez le type de la nouvelle propriété en Int32 (par défaut, la nouvelle propriété est de type String).

Step 14 - Pour modifier le type, ouvrez la fenêtre Propriétés et remplacez la propriété Type par Int32.

Step 15 - Dans Entity Framework Designer, cliquez avec le bouton droit sur la propriété Name, sélectionnez Convertir en énumération.

Step 16 - Dans la boîte de dialogue Ajouter un type d'énumération, entrez DepartmentNames pour le nom du type d'énumération, remplacez le type sous-jacent par Int32, puis ajoutez les membres suivants au type: Physique, Chimie, Informatique et Économie.

Step 17 - Cliquez sur Ok.

Si vous passez à la fenêtre Navigateur de modèles, vous verrez que le type a également été ajouté au nœud Enum Types.

Générons une base de données à partir du modèle en suivant toutes les étapes mentionnées dans le chapitre sur l'approche Model First.

Step 1 - Cliquez avec le bouton droit sur la surface Entity Designer et sélectionnez Générer la base de données à partir du modèle.

La boîte de dialogue Choisissez votre connexion de données de l'assistant de génération de base de données s'affiche.

Step 2 - Cliquez sur le bouton Nouvelle connexion.

Step 3 - Entrez le nom du serveur et EnumDemo pour la base de données et cliquez sur OK.

Step 4 - Une boîte de dialogue vous demandant si vous souhaitez créer une nouvelle base de données apparaîtra, cliquez sur Oui.

Step 5- Cliquez sur Suivant et l'assistant de création de base de données génère un langage de définition de données (DDL) pour créer une base de données. Cliquez maintenant sur Terminer.

Step 6 - Cliquez avec le bouton droit sur l'éditeur T-SQL et sélectionnez Exécuter.

Step 7 - Pour afficher le schéma généré, cliquez avec le bouton droit sur le nom de la base de données dans l'Explorateur d'objets SQL Server et sélectionnez Actualiser.

Vous verrez la table Départements dans la base de données.

Jetons un coup d'œil à l'exemple suivant dans lequel de nouveaux objets Department au contexte sont ajoutés et enregistrés. Et puis récupérez le département informatique.

class Program {

   static void Main(string[] args) {

      using (var context = new EFEnumModelContainer()) {

         context.Departments.Add(new Department { DeptName = DepartmentNames.Physics});
         context.Departments.Add(new Department { DeptName = DepartmentNames.Computer});
         context.Departments.Add(new Department { DeptName = DepartmentNames.Chemistry});
         context.Departments.Add(new Department { DeptName = DepartmentNames.Economics});

         context.SaveChanges();

         var department = (
            from d in context.Departments
            where d.DeptName == DepartmentNames.Computer
            select d
         ).FirstOrDefault();

         Console.WriteLine(
            "Department ID: {0}, Department Name: {1}", 
               department.DeptID, department.DeptName
         );

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

Department ID: 2, Department Name: Computer

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Asynchronous programmingimplique l'exécution d'opérations en arrière-plan afin que le thread principal puisse continuer ses propres opérations. De cette façon, le thread principal peut garder l'interface utilisateur réactive pendant que le thread d'arrière-plan traite la tâche en cours.

  • Entity Framework 6.0 prend en charge les opérations asynchrones pour l'interrogation et l'enregistrement des données.

  • Les opérations asynchrones peuvent aider votre application des manières suivantes:

    • Rendez votre application plus réactive aux interactions des utilisateurs
    • Améliorez les performances globales de votre application
  • Vous pouvez exécuter des opérations asynchrones de différentes manières. Mais les mots clés async / await ont été introduits dans .NET Framework 4.5, ce qui simplifie votre travail.

  • La seule chose que vous devez suivre est le modèle async / await comme illustré par le fragment de code suivant.

Jetons un coup d'œil à l'exemple suivant (sans utiliser async / await) dans lequel la méthode DatabaseOperations enregistre un nouvel étudiant dans la base de données, puis récupère tous les étudiants de la base de données et à la fin, un message supplémentaire est imprimé sur la console.

class Program {

   static void Main(string[] args) {
      Console.WriteLine("Database Operations Started");
      DatabaseOperations();
		
      Console.WriteLine();
      Console.WriteLine("Database Operations Completed");
      Console.WriteLine();
      Console.WriteLine("Entity Framework Tutorials");
      Console.ReadKey();
   }

   public static void DatabaseOperations() {

      using (var context = new UniContextEntities()) {

         // Create a new student and save it

         context.Students.Add(new Student {
            FirstMidName = "Akram", 
            LastName = "Khan", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())});

         Console.WriteLine("Calling SaveChanges.");
         context.SaveChanges();
         Console.WriteLine("SaveChanges completed.");

         // Query for all Students ordered by first name

         var students = (from s in context.Students
            orderby s.FirstMidName select s).ToList();

         // Write all students out to Console

         Console.WriteLine();
         Console.WriteLine("All Student:");

         foreach (var student in students) {
            string name = student.FirstMidName + " " + student.LastName;
            Console.WriteLine(" " + name);
         }
      }
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

Calling SaveChanges.
SaveChanges completed.
All Student:
Akram Khan
Ali Khan
Ali Alexander
Arturo Anand
Bill Gates
Gytis Barzdukas
Laura  Nornan
Meredith fllonso
Nino Olioetto
Peggy Justice
Yan Li

Entity Framework Tutorials

Utilisons les nouveaux mots-clés async et await et apportons les modifications suivantes à Program.cs

  • Ajoutez l'espace de noms System.Data.Entity qui donnera les méthodes d'extension asynchrone EF.

  • Ajoutez l'espace de noms System.Threading.Tasks qui nous permettra d'utiliser le type de tâche.

  • Mettre à jour DatabaseOperations être marqué comme async et renvoyer un Task.

  • Appelez la version Async de SaveChanges et attendez son achèvement.

  • Appelez la version Async de ToList et attendez le résultat.

class Program {

   static void Main(string[] args) {
      var task = DatabaseOperations();
      Console.WriteLine();
      Console.WriteLine("Entity Framework Tutorials");
      task.Wait();
      Console.ReadKey();
   }

   public static async Task DatabaseOperations() {

      using (var context = new UniContextEntities()) {

         // Create a new blog and save it

         context.Students.Add(new Student {
            FirstMidName = "Salman", 
            LastName = "Khan", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())});

         Console.WriteLine("Calling SaveChanges.");
         await context.SaveChangesAsync();
         Console.WriteLine("SaveChanges completed.");

         // Query for all Students ordered by first name

         var students = await (from s in context.Students 
            orderby s.FirstMidName select s).ToListAsync();

         // Write all students out to Console

         Console.WriteLine();
         Console.WriteLine("All Student:");

         foreach (var student in students) {
            string name = student.FirstMidName + " " + student.LastName; 
            Console.WriteLine(" " + name);
         }
      }
   }
}

Lors de l'exécution, il produira la sortie suivante.

Calling SaveChanges.
Entity Framework Tutorials
SaveChanges completed.
All Student:
Akram Khan
Ali Khan
Ali Alexander
Arturo Anand
Bill Gates
Gytis Barzdukas
Laura  Nornan
Meredith fllonso
Nino Olioetto
Peggy Justice
Salman Khan
Yan Li

Maintenant que le code est asynchrone, vous pouvez observer un flux d'exécution différent de votre programme.

  • SaveChanges commence à pousser le nouvel étudiant vers la base de données, puis la méthode DatabaseOperations retourne (même si elle n'a pas fini de s'exécuter) et le déroulement du programme dans la méthode Main continue.

  • Le message est ensuite écrit sur la console.

  • Le thread géré est bloqué sur l'appel Attendre jusqu'à la fin de l'opération de base de données. Une fois terminé, le reste de nos opérations de base de données sera exécuté.

  • SaveChanges se termine.

  • Récupéré tous les étudiants de la base de données et est écrit dans la console.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Entity Framework vous permet désormais de bénéficier d'Entity Framework sans forcer chaque partie de votre application à être consciente d'Entity Framework, séparant les entités de l'infrastructure. Vous pouvez créer des classes qui peuvent se concentrer sur leurs règles métier sans tenir compte de la façon dont elles sont persistantes (où les données sont stockées et comment les données vont et viennent entre vos objets).

Création d'entités ignorantes persistantes

Le paragraphe précédent décrit une méthode qui n'a aucune connaissance intime de la source des données qu'elle consomme. Cela met en évidence l'essence de l'ignorance de la persistance, c'est-à-dire lorsque vos classes et beaucoup de nos couches d'application autour d'elles ne se soucient pas de la façon dont les données sont stockées.

  • Dans la version .NET 3.5 d'Entity Framework, si vous vouliez utiliser des classes préexistantes, vous deviez les modifier en les forçant à dériver d'EntityObject.

  • Dans .NET 4, ce n'est plus nécessaire. Vous n'avez pas à modifier vos entités pour qu'elles participent aux opérations Entity Framework.

  • Cela nous permet de créer des applications qui englobent le couplage lâche et la séparation des préoccupations.

  • Avec ces modèles de codage, vos classes ne sont concernées que par leurs propres tâches et, de nombreuses couches de votre application, y compris l'interface utilisateur, n'ont aucune dépendance à la logique externe, comme les API Entity Framework, mais ces API externes sont capables d'interagir avec notre entités.

Il existe 2 façons (connectée et déconnectée) de conserver une entité avec Entity Framework. Les deux voies ont leur propre importance. Dans le cas d'un scénario connecté, les changements sont suivis par le contexte, mais dans le cas d'un scénario déconnecté, nous devons informer le contexte sur l'état de l'entité.

Scénarios connectés

Le scénario connecté est lorsqu'une entité est extraite de la base de données et modifiée dans le même contexte. Pour un scénario connecté, supposons que nous ayons un service Windows et que nous effectuions des opérations commerciales avec cette entité afin d'ouvrir le contexte, de parcourir toutes les entités, de faire nos opérations commerciales, puis d'enregistrer les modifications dans le même contexte que nous ouvert au début.

Jetons un coup d'œil à l'exemple suivant dans lequel les étudiants sont extraits de la base de données et mettent à jour le prénom des étudiants, puis enregistrent les modifications dans la base de données.

class Program {

   static void Main(string[] args) {

      using (var context = new MyContext()) {

         var studentList = context.Students.ToList();

         foreach (var stdnt in studentList) {
            stdnt.FirstMidName = "Edited " + stdnt.FirstMidName;
         }

         context.SaveChanges();

         //// Display all Students from the database

         var students = (from s in context.Students
            orderby s.FirstMidName select s).ToList<Student>();

         Console.WriteLine("Retrieve all Students from the database:");

         foreach (var stdnt in students) {
            string name = stdnt.FirstMidName + " " + stdnt.LastName;
            Console.WriteLine("ID: {0}, Name: {1}", stdnt.ID, name);
         }

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante et vous verrez que le mot édité est attaché avant le prénom, comme indiqué dans la sortie suivante.

Retrieve all Students from the database: 
ID: 1, Name: Edited Edited Alain Bomer 
ID: 2, Name: Edited Edited Mark Upston

Scénarios déconnectés

Le scénario déconnecté se produit lorsqu'une entité est extraite de la base de données et modifiée dans un contexte différent. Supposons que nous voulions afficher des données dans une couche de présentation et que nous utilisions une application à n niveaux, il serait donc préférable d'ouvrir le contexte, de récupérer les données et enfin de fermer le contexte. Puisque ici nous avons récupéré les données et fermé le contexte, les entités que nous avons récupérées ne sont plus suivies et c'est le scénario déconnecté.

Jetons un coup d'œil au code suivant dans lequel une nouvelle entité Student déconnectée est ajoutée à un contexte à l'aide de la méthode Add.

class Program {

   static void Main(string[] args) {

      var student = new Student {
         ID = 1001, 
         FirstMidName = "Wasim", 
         LastName = "Akram", 
         EnrollmentDate = DateTime.Parse( DateTime.Today.ToString())
      };

      using (var context = new MyContext()) {

         context.Students.Add(student);
         context.SaveChanges();

         //// Display all Students from the database

         var students = (from s in context.Students 
            orderby s.FirstMidName select s).ToList<Student>();

         Console.WriteLine("Retrieve all Students from the database:");

         foreach (var stdnt in students) {
            string name = stdnt.FirstMidName + " " + stdnt.LastName;
            Console.WriteLine("ID: {0}, Name: {1}", stdnt.ID, name);
         }

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante.

Retrieve all Students from the database:
ID: 1, Name: Edited Edited Edited Alain Bomer
ID: 2, Name: Edited Edited Edited Mark Upston
ID: 3, Name: Wasim Akram

LINQ vers les entités

L'un des concepts les plus importants pour comprendre LINQ to Entities est qu'il s'agit d'un langage déclaratif. L'accent est mis sur la définition des informations dont vous avez besoin, plutôt que sur la manière de les obtenir.

  • Cela signifie que vous pouvez passer plus de temps à travailler avec des données et moins de temps à essayer de comprendre le code sous-jacent requis pour effectuer des tâches telles que l'accès à la base de données.

  • Il est important de comprendre que les langages déclaratifs ne suppriment en fait aucun contrôle du développeur, mais cela aide le développeur à concentrer son attention sur ce qui est important.

Mots-clés essentiels LINQ to Entities

Il est important de connaître les mots-clés de base utilisés pour créer une requête LINQ. Il n'y a que quelques mots clés à retenir, mais vous pouvez les combiner de différentes manières pour obtenir des résultats spécifiques. La liste suivante contient ces mots clés de base et fournit une description simple de chacun d'eux.

Sr. No. Mot-clé et description
1

Ascending

Spécifie qu'une opération de tri a lieu du plus petit (ou le plus bas) élément d'une plage à l'élément le plus élevé d'une plage. Il s'agit normalement du paramètre par défaut. Par exemple, lors d'un tri alphabétique, le tri serait compris entre A et Z.

2

By

Spécifie le champ ou l'expression utilisé pour implémenter un regroupement. Le champ ou l'expression définit une clé utilisée pour effectuer la tâche de regroupement.

3

Descending

Spécifie qu'une opération de tri a lieu du plus grand (ou le plus élevé) élément d'une plage à l'élément le plus bas d'une plage. Par exemple, lors d'un tri alphabétique, le tri serait compris entre Z et A.

4

Equals

Utilisé entre les clauses gauche et droite d'une instruction de jointure pour joindre la source de données contextuelles principale à la source de données contextuelles secondaire. Le champ ou l'expression à gauche du mot clé equals spécifie la source de données principale, tandis que le champ ou l'expression à droite du mot clé equals spécifie la source de données secondaire.

5

From

Spécifie la source de données utilisée pour obtenir les informations requises et définit une variable de plage. Cette variable a le même objectif qu'une variable utilisée pour l'itération dans une boucle.

6

Group

Organise la sortie en groupes à l'aide de la valeur de clé que vous spécifiez. Utilisez plusieurs clauses de groupe pour créer plusieurs niveaux d'organisation de sortie. L'ordre des clauses de groupe détermine la profondeur à laquelle une valeur de clé particulière apparaît dans l'ordre de regroupement. Vous combinez ce mot-clé avec by pour créer un contexte spécifique.

sept

In

Utilisé de plusieurs manières. Dans ce cas, le mot-clé détermine la source de base de données contextuelle utilisée pour une requête. Lorsque vous travaillez avec une jointure, le mot clé in est utilisé pour chaque source de base de données contextuelle utilisée pour la jointure.

8

Into

Spécifie un identificateur que vous pouvez utiliser comme référence pour les clauses de requête LINQ telles que join, group et select.

9

Join

Crée une seule source de données à partir de deux sources de données associées, comme dans une configuration maître / détail. Une jointure peut spécifier une jointure interne, de groupe ou externe gauche, avec la jointure interne par défaut. Vous pouvez en savoir plus sur les jointures sur msdn.microsoft.com

dix

Let

Définit une variable de plage que vous pouvez utiliser pour stocker les résultats de sous-expression dans une expression de requête. En règle générale, la variable de plage est utilisée pour fournir une sortie énumérée supplémentaire ou pour augmenter l'efficacité d'une requête (de sorte qu'une tâche particulière, telle que la recherche de la valeur minuscule d'une chaîne, ne doive pas être effectuée plus d'une fois).

11

On

Spécifie le champ ou l'expression utilisé pour implémenter une jointure. Le champ ou l'expression définit un élément commun aux deux sources de données contextuelles.

12

Orderby

Crée un ordre de tri pour la requête. Vous pouvez ajouter le mot-clé croissant ou décroissant pour contrôler l'ordre du tri. Utilisez plusieurs clauses orderby pour créer plusieurs niveaux de tri. L'ordre des clauses orderby détermine l'ordre dans lequel les expressions de tri sont traitées, donc l'utilisation d'un ordre différent entraînera une sortie différente.

13

Where

Définit ce que LINQ doit extraire de la source de données. Vous utilisez une ou plusieurs expressions booléennes pour définir les spécificités des éléments à récupérer. Les expressions booléennes sont séparées les unes des autres à l'aide des symboles && (AND) et || (OR) opérateurs.

14

Select

Détermine la sortie de la requête LINQ en spécifiant les informations à renvoyer. Cette instruction définit le type de données des éléments renvoyés par LINQ pendant le processus d'itération.

Projection

Les requêtes de projection améliorent l'efficacité de votre application, en ne récupérant que des champs spécifiques de votre base de données.

  • Une fois que vous avez les données, vous voudrez peut-être les projeter ou les filtrer selon les besoins pour façonner les données avant la sortie.

  • La tâche principale de toute expression LINQ to Entities est d'obtenir des données et de les fournir en sortie.

La section «Développement de requêtes LINQ to Entities» de ce chapitre présente les techniques permettant d'effectuer cette tâche de base.

Jetons un coup d'œil au code suivant dans lequel la liste des étudiants sera récupérée.

using (var context = new UniContextEntities()) {

   var studentList = from s in context.Students select s;

   foreach (var student in studentList) {
      string name = student.FirstMidName + " " + student.LastName;
      Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
   }
}

Objet unique

Pour récupérer un seul objet étudiant, vous pouvez utiliser les méthodes énumérables First () ou FirstOrDefault qui retourne le premier élément d'une séquence. La différence entre First et FirstOrDefault est que First () lèvera une exception s'il n'y a pas de données de résultat pour les critères fournis alors que FirstOrDefault () renvoie la valeur par défaut null, s'il n'y a pas de données de résultat. Dans l'extrait de code ci-dessous, le premier étudiant de la liste sera récupéré dont le prénom est Ali.

using (var context = new UniContextEntities()) {

   var student = (from s in context.Students where s.FirstMidName 
      == "Ali" select s).FirstOrDefault<Student>();

   string name = student.FirstMidName + " " + student.LastName;
   Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
}

Vous pouvez également utiliser Single () ou SingleOrDefault pour obtenir un seul objet étudiant qui renvoie un seul élément spécifique d'une séquence. Dans l'exemple suivant, un seul étudiant est récupéré dont l'ID est 2.

using (var context = new UniContextEntities()) {

   var student = (from s in context.Students where s.ID 
      == 2 select s).SingleOrDefault<Student>();
   string name = student.FirstMidName + " " + student.LastName;
	
   Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
   Console.ReadKey();
}

Liste des objets

Si vous souhaitez récupérer la liste des étudiants dont le prénom est Ali, vous pouvez utiliser la méthode énumérable ToList ().

using (var context = new UniContextEntities()) {

   var studentList = (from s in context.Students where s.FirstMidName 
      == "Ali" select s).ToList();

   foreach (var student in studentList) {
      string name = student.FirstMidName + " " + student.LastName;
      Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
   }

   Console.ReadKey();
}

Ordre

Pour récupérer des données / liste dans un ordre particulier, vous pouvez utiliser le mot-clé orderby. Dans le code suivant, la liste d'extraits de l'élève sera récupérée par ordre croissant.

using (var context = new UniContextEntities()) {

   var studentList = (from s in context.Students orderby
      s.FirstMidName ascending select s).ToList();

   foreach (var student in studentList) {
      string name = student.FirstMidName + " " + student.LastName;
      Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
   }

   Console.ReadKey();
}

Requête Standard Vs Projection Entity Framework

Supposons que vous ayez un modèle Student contenant l'ID, FirstMidName, LastName et EnrollmentDate. Si vous souhaitez renvoyer une liste d'étudiants, une requête standard renverra tous les champs. Mais si vous souhaitez uniquement obtenir une liste d'étudiants contenant les champs ID, FirstMidName et LastName. C'est là que vous devez utiliser une requête de projection. Voici un exemple simple de requête de projection.

using (var context = new UniContextEntities()) {

   var studentList = from s in context.Students
      orderby s.FirstMidName ascending
      where s.FirstMidName == "Ali"

   select new {s.ID, s.FirstMidName, s.LastName};

   foreach (var student in studentList) {
      string name = student.FirstMidName + " " + student.LastName;
      Console.WriteLine("ID : {0}, Name: {1}", student.ID, name);
   }

   Console.ReadKey();
}

La requête de projection ci-dessus exclut le champ EnrollmentDate. Cela rendra votre demande beaucoup plus rapide.

Dans Entity Framework 6.0, une nouvelle fonctionnalité est introduite, appelée Logging SQL. Tout en travaillant avec Entity Framework, il envoie des commandes ou une requête SQL équivalente à la base de données pour effectuer des opérations CRUD (Créer, Lire, Mettre à jour et Supprimer).

  • Cette fonctionnalité d'Entity Framework consiste à capturer une requête SQL équivalente générée par Entity Framework en interne et à la fournir en sortie.

  • Avant Entity Framework 6, chaque fois qu'il était nécessaire de tracer des requêtes et des commandes de base de données, le développeur n'avait d'autre choix que d'utiliser un utilitaire de traçage tiers ou un outil de traçage de base de données.

  • Dans Entity Framework 6, cette nouvelle fonctionnalité fournit un moyen simple en journalisant toutes les opérations effectuées par Entity Framework.

  • Toutes les activités effectuées par Entity Framework sont enregistrées à l'aide de DbContext.Database.Log.

Jetons un coup d'œil au code suivant dans lequel un nouvel étudiant est ajouté à la base de données.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         context.Database.Log = Console.Write;

         // Create a new student and save it

         context.Students.Add(new Student {
            FirstMidName = "Salman", 
            LastName = "Khan", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         });

         context.SaveChanges();
         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante, qui est en fait le journal de toutes les activités effectuées par EF dans le code ci-dessus.

Opened connection at 10/28/2015 6:27:35 PM +05:00
Started transaction at 10/28/2015 6:27:35 PM +05:00
INSERT [dbo].[Student]([LastName], [FirstMidName], [EnrollmentDate])
VALUES (@0, @1, @2)
SELECT [ID]
FROM [dbo].[Student]
WHERE @@ROWCOUNT > 0 AND [ID] = scope_identity()
-- @0: 'Khan' (Type = String, Size = -1)
-- @1: 'Salman' (Type = String, Size = -1)
-- @2: '10/28/2015 12:00:00 AM' (Type = DateTime)
-- Executing at 10/28/2015 6:27:35 PM +05:00
-- Completed in 5 ms with result: SqlDataReader
Committed transaction at 10/28/2015 6:27:35 PM +05:00
Closed connection at 10/28/2015 6:27:35 PM +05:00

Lorsque la propriété Log est définie, les activités suivantes sont enregistrées -

  • SQL pour tous les types de commandes, par exemple les requêtes, y compris les insertions, les mises à jour et les suppressions générées dans le cadre de SaveChanges

  • Parameters

  • Si la commande est exécutée de manière asynchrone ou non

  • Un horodatage indiquant quand la commande a commencé à s'exécuter

  • La commande s'est terminée avec succès ou a échoué

  • Quelques indications sur la valeur du résultat

  • Le temps approximatif nécessaire pour exécuter la commande

Connexion à un autre endroit

Si vous disposez déjà d'un cadre de journalisation et qu'il définit une méthode de journalisation, vous pouvez également le connecter à un autre endroit.

Jetons un coup d'œil à l'exemple suivant dans lequel nous avons une autre classe MyLogger.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         context.Database.Log = s ⇒ MyLogger.Log("EFLoggingDemo", s);

         // Create a new student and save it

         context.Students.Add(new Student {
            FirstMidName = "Salman", 
            LastName = "Khan", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         });

         context.SaveChanges();
         Console.ReadKey();
      }
   }
}

public class MyLogger {

   public static void Log(string application, string message) {
      Console.WriteLine("Application: {0}, EF Message: {1} ",application, message);
   }
}

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans Entity Framework 6.0, il existe une autre nouvelle fonctionnalité appelée Interceptorou Interception. Le code d'interception est construit autour du concept deinterception interfaces. Par exemple, l'interface IDbCommandInterceptor définit des méthodes qui sont appelées avant qu'EF n'appelle ExecuteNonQuery, ExecuteScalar, ExecuteReader et les méthodes associées.

  • Entity Framework peut vraiment briller en utilisant l'interception. En utilisant cette approche, vous pouvez capturer beaucoup plus d'informations de manière transitoire sans avoir à désorganiser votre code.

  • Pour implémenter cela, vous devez créer votre propre intercepteur personnalisé et l'enregistrer en conséquence.

  • Une fois qu'une classe qui implémente l'interface IDbCommandInterceptor a été créée, elle peut être inscrite avec Entity Framework à l'aide de la classe DbInterception.

  • L'interface IDbCommandInterceptor a six méthodes et vous devez implémenter toutes ces méthodes. Voici la mise en œuvre de base de ces méthodes.

Jetons un coup d'œil au code suivant dans lequel l'interface IDbCommandInterceptor est implémentée.

public class MyCommandInterceptor : IDbCommandInterceptor {

   public static void Log(string comm, string message) {
      Console.WriteLine("Intercepted: {0}, Command Text: {1} ", comm, message);
   }

   public void NonQueryExecuted(DbCommand command, 
      DbCommandInterceptionContext<int> interceptionContext) {
         Log("NonQueryExecuted: ", command.CommandText);
   }

   public void NonQueryExecuting(DbCommand command, 
      DbCommandInterceptionContext<int> interceptionContext) {
         Log("NonQueryExecuting: ", command.CommandText);
   }

   public void ReaderExecuted(DbCommand command, 
      DbCommandInterceptionContext<DbDataReader> interceptionContext) {
         Log("ReaderExecuted: ", command.CommandText);
   }

   public void ReaderExecuting(DbCommand command, 
      DbCommandInterceptionContext<DbDataReader> interceptionContext) {
         Log("ReaderExecuting: ", command.CommandText);
   }

   public void ScalarExecuted(DbCommand command, 
      DbCommandInterceptionContext<object> interceptionContext) {
         Log("ScalarExecuted: ", command.CommandText);
   }

   public void ScalarExecuting(DbCommand command, 
      DbCommandInterceptionContext<object> interceptionContext) {
         Log("ScalarExecuting: ", command.CommandText);
   }

}

Enregistrement des intercepteurs

Une fois qu'une classe qui implémente une ou plusieurs des interfaces d'interception a été créée, elle peut être inscrite auprès d'EF à l'aide de la classe DbInterception, comme indiqué dans le code suivant.

DbInterception.Add(new MyCommandInterceptor());

Les intercepteurs peuvent également être enregistrés au niveau du domaine d'application à l'aide de la configuration basée sur le code DbConfiguration, comme indiqué dans le code suivant.

public class MyDBConfiguration : DbConfiguration {

   public MyDBConfiguration() {
      DbInterception.Add(new MyCommandInterceptor());
   }
}

Vous pouvez également configurer le fichier de configuration de l'intercepteur en utilisant le code -

<entityFramework>
   <interceptors>
      <interceptor type = "EFInterceptDemo.MyCommandInterceptor, EFInterceptDemo"/>
   </interceptors>
</entityFramework>

La prise en charge du type spatial a été introduite dans Entity Framework 5. Un ensemble d'opérateurs est également inclus pour permettre aux requêtes d'analyser les données spatiales. Par exemple, une requête peut filtrer en fonction de la distance entre deux emplacements géographiques.

  • Entity Framework permettra à de nouveaux types de données spatiales d'être exposés en tant que propriétés sur vos classes et de les mapper aux colonnes spatiales de votre base de données.

  • Vous pourrez également écrire des requêtes LINQ qui utilisent les opérateurs spatiaux pour filtrer, trier et regrouper en fonction des calculs spatiaux effectués dans la base de données.

Il existe deux principaux types de données spatiales -

  • Le type de données géographie stocke des données ellipsoïdales, par exemple, les coordonnées GPS de latitude et de longitude.

  • Le type de données geometry représente le système de coordonnées euclidien (plat).

Jetons un coup d'œil à l'exemple suivant de Cricket Ground.

Step 1 - Créer un nouveau projet à partir de l'option de menu Fichier → Nouveau → Projet.

Step 2 - Dans le volet gauche, sélectionnez l'application console.

Step 3 - Cliquez avec le bouton droit sur le nom du projet et sélectionnez Gérer les packages NuGet…

Step 4 - Installez Entity Framework.

Step 5 - Ajoutez une référence à l'assembly System.Data.Entity et ajoutez également l'instruction System.Data.Spatial using pour les types de données spatiales.

Step 6 - Ajoutez la classe suivante dans le fichier Program.cs.

public class CricketGround {
   public int ID { get; set; }
   public string Name { get; set; }
   public DbGeography Location { get; set; }
}

Step 7 - En plus de définir des entités, vous devez définir une classe qui dérive de DbContext et expose les propriétés DbSet <TEntity>.

Dans Program.cs, ajoutez la définition de contexte.

public partial class CricketGroundContext : DbContext {
   public DbSet<CricketGround> CricketGrounds { get; set; }
}

Step 8 - Ajoutez le code suivant dans la fonction Main, qui ajoutera deux nouveaux objets CricketGround au contexte.

class Program {

   static void Main(string[] args) {

      using (var context = new CricketGroundContext()) {

         context.CricketGrounds.Add(new CricketGround() {
            Name = "Shalimar Cricket Ground", 
            Location = DbGeography.FromText("POINT(-122.336106 47.605049)"), 
         });

         context.CricketGrounds.Add(new CricketGround() {
            Name = "Marghazar Stadium", Location = DbGeography
               .FromText("POINT(-122.335197 47.646711)"), 
         });

         context.SaveChanges();

         var myLocation = DbGeography.FromText("POINT(-122.296623 47.640405)");

         var cricketGround = (from cg in context.CricketGrounds
            orderby cg.Location.Distance(myLocation) select cg).FirstOrDefault();

         Console.WriteLine("The closest Cricket Ground to you is: {0}.", cricketGround.Name);
      }
   }
}

Les propriétés spatiales sont initialisées à l'aide de la méthode DbGeography.FromText. Le point géographique représenté par WellKnownText est passé à la méthode, puis enregistre les données. Après cela, l'objet CricketGround sera récupéré là où son emplacement est le plus proche de l'emplacement spécifié.

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

The closest Cricket Ground to you is: Marghazar Stadium

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

L'héritage permet de créer des modèles complexes qui reflètent mieux la façon dont les développeurs pensent et réduisent également le travail requis pour interagir avec ces modèles. L'héritage utilisé avec des entités a le même objectif que l'héritage utilisé avec des classes, de sorte que les développeurs connaissent déjà les bases du fonctionnement de cette fonctionnalité.

Jetons un coup d'œil à l'exemple suivant et en créant un nouveau projet d'application console.

Step 1 - Ajoutez le modèle de données d'entité ADO.NET en cliquant avec le bouton droit sur le nom du projet et sélectionnez Ajouter → Nouvel élément…

Step 2 - Ajoutez une entité et nommez-la Person en suivant toutes les étapes mentionnées dans le chapitre Approche Model First.

Step 3 - Ajoutez des propriétés scalaires comme indiqué dans l'image suivante.

Step 4 - Nous ajouterons deux autres entités Student et Teacher, qui héritera des propriétés de Person Table.

Step 5 - Ajoutez maintenant une entité Etudiant et sélectionnez Personne dans la liste déroulante Type de base, comme indiqué dans l'image suivante.

Step 6 - De même, ajoutez une entité Enseignant.

Step 7 - Ajoutez maintenant la propriété scalaire EnrollmentDate à l'entité Student et la propriété HireDate à l'entité Teacher.

Step 8 - Allons-y et générons la base de données.

Step 9 - Faites un clic droit sur l'aire de conception et sélectionnez Générer la base de données à partir du modèle…

Step 10- Pour créer une nouvelle base de données, cliquez sur Nouvelle connexion… La boîte de dialogue suivante s'ouvre. Cliquez sur OK.

Step 11- Cliquez sur Terminer. Cela ajoutera le fichier * .edmx.sql dans le projet. Vous pouvez exécuter des scripts DDL dans Visual Studio en ouvrant le fichier .sql. Maintenant, cliquez avec le bouton droit de la souris et sélectionnez Exécuter.

Step 12 - Allez dans l'explorateur de serveur, vous verrez que la base de données est créée avec trois tables qui sont spécifiées.

Step 13 - Vous pouvez également voir que les classes de domaine suivantes sont également générées automatiquement.

public partial class Person {
   public int ID { get; set; }
   public string FirstMidName { get; set; }
   public string LastName { get; set; }
}

public partial class Student : Person {
   public System.DateTime EnrollmentDate { get; set; }
}

public partial class Teacher : Person {
   public System.DateTime HireDate { get; set; }
}

Voici la classe Context.

public partial class InheritanceModelContainer : DbContext {

   public InheritanceModelContainer() : 
      base("name = InheritanceModelContainer") {}

   protected override void OnModelCreating(DbModelBuilder modelBuilder) {
      throw new UnintentionalCodeFirstException();
   }

   public virtual DbSet<Person> People { get; set; }
}

Ajoutons des étudiants et des enseignants à la base de données, puis récupérons-les de la base de données.

class Program {

   static void Main(string[] args) {

      using (var context = new InheritanceModelContainer()) {

         var student = new Student {
            FirstMidName = "Meredith", 
            LastName = "Alonso", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         };

         context.People.Add(student);

         var student1 = new Student {
            FirstMidName = "Arturo", 
            LastName = "Anand", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         };

         context.People.Add(student1);

         var techaer = new Teacher {
            FirstMidName = "Peggy", 
            LastName = "Justice", 
            HireDate = DateTime.Parse(DateTime.Today.ToString())
         };

         context.People.Add(techaer);

         var techaer1 = new Teacher {
            FirstMidName = "Yan", 
            LastName = "Li", 
            HireDate = DateTime.Parse(DateTime.Today.ToString())
         };

         context.People.Add(techaer1);
         context.SaveChanges();
      }
   }
}

Les étudiants et les enseignants sont ajoutés dans la base de données. NTour récupérer les élèves et l'enseignant, leOfType La méthode doit être utilisée, ce qui renverra l'étudiant et l'enseignant liés au département spécifié.

Console.WriteLine("All students in database"); 
Console.WriteLine("");

foreach (var student in context.People.OfType<Student>()) {
   string name = student.FirstMidName + " " + student.LastName;
   Console.WriteLine("ID: {0}, Name: {1}, \tEnrollment Date {2} ", 
      student.ID, name, student.EnrollmentDate.ToString());
}

Console.WriteLine("");
Console.WriteLine("************************************************************ *****");
Console.WriteLine("");
Console.WriteLine("All teachers in database");
Console.WriteLine("");

foreach (var teacher in context.People.OfType<Teacher>()) {
   string name = teacher.FirstMidName + " " + teacher.LastName;
   Console.WriteLine("ID: {0}, Name: {1}, \tHireDate {2} ", 
      teacher.ID, name, teacher.HireDate.ToString()); 
}

Console.WriteLine("");
Console.WriteLine("************************************************************ *****");
Console.ReadKey();

Dans la première requête, lorsque vous utilisez OfType <Student> (), vous ne pourrez pas accéder à HireDate car la propriété HireDate fait partie de Teacher Entity et de même, la propriété EnrollmentDate ne sera pas accessible lorsque vous utilisez OfType <Teacher> ()

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante -

All students in database
ID: 1, Name: Meredith Alonso,   Enrollment Date 10/30/2015 12:00:00 AM
ID: 2, Name: Arturo Anand,      Enrollment Date 10/30/2015 12:00:00 AM
*****************************************************************  
All teachers in database
ID: 3, Name: Peggy Justice,     HireDate 10/30/2015 12:00:00 AM
ID: 4, Name: Yan Li,    HireDate 10/30/2015 12:00:00 AM
*****************************************************************

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans Entity Framework 5 et les versions précédentes d'Entity Framework, le code était divisé entre les bibliothèques principales (principalement System.Data.Entity.dll) fournies dans le cadre du .NET Framework, et les bibliothèques supplémentaires (principalement EntityFramework.dll) étaient distribuées et expédié à l'aide de NuGet, comme illustré dans le diagramme suivant.

Dans Entity Framework 6, les API principales qui faisaient auparavant partie de .NET Framework sont également fournies et distribuées dans le cadre du package NuGet.

Cela était nécessaire pour permettre à Entity Framework d'être rendu open source. Cependant, par conséquent, les applications devront être reconstruites chaque fois qu'il est nécessaire de migrer ou de mettre à niveau votre application des anciennes versions d'Entity Framework vers EF 6.

Le processus de migration est simple si votre application utilise DbContext, qui a été fourni dans EF 4.1 et versions ultérieures. Mais si votre application est ObjectContext, peu de travail supplémentaire est nécessaire.

Jetons un coup d'œil aux étapes suivantes que vous devez suivre pour mettre à niveau une application existante vers EF6.

Step 1 - La première étape consiste à cibler .NET Framework 4.5.2, puis cliquez avec le bouton droit sur votre projet et sélectionnez les propriétés.

Step 2 - Cliquez à nouveau avec le bouton droit sur votre projet et sélectionnez Gérer les packages NuGet ...

Step 3- Sous l'onglet En ligne, sélectionnez EntityFramework et cliquez sur Installer. Assurez-vous que les références d'assembly à System.Data.Entity.dll sont supprimées.

Lorsque vous installez le package EF6 NuGet, il doit supprimer automatiquement toutes les références à System.Data.Entity de votre projet pour vous.

Step 4 - Si vous avez un modèle créé avec EF Designer, vous devrez également mettre à jour les modèles de génération de code pour générer du code compatible EF6.

Step 5 - Dans votre Explorateur de solutions sous votre fichier edmx, supprimez les modèles de génération de code existants qui seront généralement nommés <edmx_file_name> .tt et <edmx_file_name> .Context.tt.

Step 6 - Ouvrez votre modèle dans EF Designer, cliquez avec le bouton droit sur l'aire de conception et sélectionnez Ajouter un élément de génération de code ...

Step 7 - Ajoutez le modèle de génération de code EF 6.x approprié.

Il générera également automatiquement du code compatible EF6.

Si vos applications utilisent EF 4.1 ou version ultérieure, vous n'aurez pas besoin de modifier quoi que ce soit dans le code, car les espaces de noms pour les types DbContext et Code First n'ont pas changé.

Mais si votre application utilise une ancienne version d'Entity Framework, des types tels que ObjectContext qui se trouvaient auparavant dans System.Data.Entity.dll ont été déplacés vers de nouveaux espaces de noms.

Step 8 - Vous devrez mettre à jour vos directives using ou Import pour construire avec EF6.

La règle générale pour les modifications d'espace de noms est que tout type de System.Data. * Est déplacé vers System.Data.Entity.Core. *. Voici quelques-uns d'entre eux -

  • System.Data.EntityException ⇒ System.Data.Entity.Core.EntityException
  • System.Data.Objects.ObjectContext ⇒ System.Data.Entity.Core.Objects.ObjectContext;
  • System.Data.Objects.DataClasses.RelationshipManager ⇒ System.Data.Entity.Core.Objects.DataClasses.RelationshipManager;

Certains types se trouvent dans les espaces de noms Core car ils ne sont pas utilisés directement pour la plupart des applications basées sur DbContext.

  • System.Data.EntityState ⇒ System.Data.Entity.EntityState
  • System.Data.Objects.DataClasses.EdmFunctionAttribute ⇒ System.Data.Entity.DbFunctionAttribute

Votre projet Entity Framework existant fonctionnera dans Entity Framework 6.0 sans aucune modification majeure.

Le chargement hâtif est le processus par lequel une requête pour un type d'entité charge également des entités associées dans le cadre de la requête. Le chargement hâtif est obtenu par l'utilisation duInclude method.

Cela signifie que la demande de données associées doit être renvoyée avec les résultats de la requête de la base de données. Il n'y a qu'une seule connexion établie à la source de données, une plus grande quantité de données est renvoyée dans la requête initiale.

Par exemple, lorsque vous interrogez des étudiants, chargez avec impatience leurs inscriptions. Les étudiants et leurs inscriptions seront récupérés en une seule requête.

Jetons un coup d'œil à l'exemple suivant dans lequel tous les étudiants avec leurs inscriptions respectives sont extraits de la base de données à l'aide du chargement hâtif.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {
         // Load all students and related enrollments
         var students = context.Students
            .Include(s ⇒ s.Enrollments).ToList();
			
         foreach (var student in students) {
            string name = student.FirstMidName + " " + student.LastName;
            Console.WriteLine("ID: {0}, Name: {1}", student.ID, name);
				
            foreach (var enrollment in student.Enrollments) {
               Console.WriteLine("Enrollment ID: {0}, Course ID: {1}", 
                  enrollment.EnrollmentID, enrollment.CourseID);
            }
         }

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante.

ID: 1, Name: Ali Alexander
       Enrollment ID: 1, Course ID: 1050
       Enrollment ID: 2, Course ID: 4022
       Enrollment ID: 3, Course ID: 4041
ID: 2, Name: Meredith Alonso
       Enrollment ID: 4, Course ID: 1045
       Enrollment ID: 5, Course ID: 3141
       Enrollment ID: 6, Course ID: 2021
ID: 3, Name: Arturo Anand
       Enrollment ID: 7, Course ID: 1050
ID: 4, Name: Gytis Barzdukas
       Enrollment ID: 8, Course ID: 1050
       Enrollment ID: 9, Course ID: 4022

Voici quelques-unes des autres formes de requêtes de chargement hâtif qui peuvent être utilisées.

// Load one Student and its related enrollments

var student1 = context.Students
   .Where(s ⇒ s.FirstMidName == "Ali")
   .Include(s ⇒ s.Enrollments).FirstOrDefault();

// Load all Students and related enrollments
// using a string to specify the relationship

var studentList = context.Students
   .Include("Enrollments").ToList();

// Load one Student and its related enrollments
// using a string to specify the relationship

var student = context.Students
   .Where(s ⇒ s.FirstMidName == "Salman")
   .Include("Enrollments").FirstOrDefault();

Niveaux multiples

Il est également possible de charger rapidement plusieurs niveaux d'entités associées. Les requêtes suivantes montrent des exemples d'étudiants, d'inscriptions et de cours.

// Load all Students, all related enrollments, and all related courses

var studentList = context.Students
   .Include(s ⇒ s.Enrollments.Select(c ⇒ c.Course)).ToList();

// Load all Students, all related enrollments, and all related courses
// using a string to specify the relationships

var students = context.Students
   .Include("Enrollments.Course").ToList();

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Le chargement différé est le processus par lequel une entité ou une collection d'entités est automatiquement chargée à partir de la base de données lors du premier accès à une propriété faisant référence à l'entité / aux entités. Le chargement différé signifie retarder le chargement des données associées, jusqu'à ce que vous en ayez spécifiquement la demande.

  • Lors de l'utilisation de types d'entités POCO, le chargement différé est obtenu en créant des instances de types de proxy dérivés, puis en remplaçant les propriétés virtuelles pour ajouter le hook de chargement.

  • Le chargement paresseux est à peu près la valeur par défaut.

  • Si vous laissez la configuration par défaut et que vous n'indiquez pas explicitement à Entity Framework dans votre requête que vous voulez autre chose que le chargement différé, alors le chargement différé est ce que vous obtiendrez.

  • Par exemple, lors de l'utilisation de la classe d'entité Student, les inscriptions associées seront chargées lors du premier accès à la propriété de navigation Inscriptions.

  • La propriété de navigation doit être définie comme publique, virtuelle. Le contexte vaNOT effectuez un chargement différé si la propriété n'est pas définie comme virtuelle.

Voici une classe Student qui contient la propriété de navigation des inscriptions.

public partial class Student {

   public Student() {
      this.Enrollments = new HashSet<Enrollment>();
   }
	
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public System.DateTime EnrollmentDate { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Jetons un coup d'œil à un exemple simple dans lequel la liste des étudiants est d'abord chargée à partir de la base de données, puis elle chargera les inscriptions d'un étudiant en particulier chaque fois que vous en aurez besoin.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         //Loading students only
         IList<Student> students = context.Students.ToList<Student>();

         foreach (var student in students) {

            string name = student.FirstMidName + " " + student.LastName;
            Console.WriteLine("ID: {0}, Name: {1}", student.ID, name);
	
            foreach (var enrollment in student.Enrollments) {
               Console.WriteLine("Enrollment ID: {0}, Course ID: {1}", 
                  enrollment.EnrollmentID, enrollment.CourseID);
            }
         }

         Console.ReadKey();
      }
   }
}

Lorsque le code ci-dessus est compilé et exécuté, vous recevrez la sortie suivante.

ID: 1, Name: Ali Alexander
       Enrollment ID: 1, Course ID: 1050
       Enrollment ID: 2, Course ID: 4022
       Enrollment ID: 3, Course ID: 4041
ID: 2, Name: Meredith Alonso
       Enrollment ID: 4, Course ID: 1045
       Enrollment ID: 5, Course ID: 3141
       Enrollment ID: 6, Course ID: 2021
ID: 3, Name: Arturo Anand
       Enrollment ID: 7, Course ID: 1050
ID: 4, Name: Gytis Barzdukas
       Enrollment ID: 8, Course ID: 1050
       Enrollment ID: 9, Course ID: 4022
ID: 5, Name: Yan Li
       Enrollment ID: 10, Course ID: 4041
ID: 6, Name: Peggy Justice
       Enrollment ID: 11, Course ID: 1045
ID: 7, Name: Laura Norman
       Enrollment ID: 12, Course ID: 3141

Désactiver le chargement différé

Le chargement paresseux et la sérialisation ne se mélangent pas bien, et si vous ne faites pas attention, vous pouvez finir par interroger toute votre base de données simplement parce que le chargement paresseux est activé. Il est recommandé de désactiver le chargement différé avant de sérialiser une entité.

Désactivation pour des propriétés de navigation spécifiques

Le chargement différé de la collection Enrollments peut être désactivé en rendant la propriété Enrollments non virtuelle, comme illustré dans l'exemple suivant.

public partial class Student { 

   public Student() { 
      this.Enrollments = new HashSet<Enrollment>(); 
   }
	
   public int ID { get; set; } 
   public string LastName { get; set; } 
   public string FirstMidName { get; set; } 
   public System.DateTime EnrollmentDate { get; set; }
   public ICollection<Enrollment> Enrollments { get; set; } 
}

Désactiver pour toutes les entités

Le chargement différé peut être désactivé pour toutes les entités du contexte en définissant un indicateur de la propriété Configuration sur false, comme illustré dans l'exemple suivant.

public partial class UniContextEntities : DbContext { 

   public UniContextEntities(): base("name=UniContextEntities") {
      this.Configuration.LazyLoadingEnabled = false;
   }
	
   protected override void OnModelCreating(DbModelBuilder modelBuilder) { 
      throw new UnintentionalCodeFirstException(); 
   } 
}

Après avoir désactivé le chargement différé, maintenant, lorsque vous exécutez à nouveau l'exemple ci-dessus, vous verrez que les inscriptions ne sont pas chargées et que seules les données des étudiants sont récupérées.

ID: 1, Name: Ali Alexander
ID: 2, Name: Meredith Alons
ID: 3, Name: Arturo Anand
ID: 4, Name: Gytis Barzduka
ID: 5, Name: Yan Li
ID: 6, Name: Peggy Justice
ID: 7, Name: Laura Norman
ID: 8, Name: Nino Olivetto

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Lorsque vous avez désactivé le chargement différé, il est toujours possible de charger paresseusement les entités associées, mais cela doit être fait avec un appel explicite.

  • Contrairement au chargement différé, il n'y a pas d'ambiguïté ou de possibilité de confusion concernant le moment où une requête est exécutée.

  • Pour ce faire, vous utilisez la méthode Load sur l'entrée de l'entité associée.

  • Pour une relation un-à-plusieurs, appelez la méthode Load sur Collection.

  • Et pour une relation un-à-un, appelez la méthode Load sur Reference.

Jetons un œil à l'exemple suivant dans lequel le chargement différé est désactivé, puis l'étudiant dont le prénom est Ali est récupéré.

Les informations sur l'étudiant sont ensuite écrites sur la console. Si vous regardez le code, les informations sur les inscriptions sont également écrites, mais l'entité Enrollments n'est pas encore chargée, donc la boucle foreach ne sera pas exécutée.

Une fois que l'entité Inscriptions est chargée explicitement, les informations sur les étudiants et les inscriptions seront désormais écrites dans la fenêtre de la console.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         context.Configuration.LazyLoadingEnabled = false;

         var student = (from s in context.Students where s.FirstMidName == 
            "Ali" select s).FirstOrDefault<Student>();

         string name = student.FirstMidName + " " + student.LastName;
         Console.WriteLine("ID: {0}, Name: {1}", student.ID, name);

         foreach (var enrollment in student.Enrollments) {
            Console.WriteLine("Enrollment ID: {0}, Course ID: {1}", 
               enrollment.EnrollmentID, enrollment.CourseID);
         }

         Console.WriteLine();
         Console.WriteLine("Explicitly loaded Enrollments");
         Console.WriteLine();

         context.Entry(student).Collection(s ⇒ s.Enrollments).Load();
         Console.WriteLine("ID: {0}, Name: {1}", student.ID, name);

         foreach (var enrollment in student.Enrollments) {
            Console.WriteLine("Enrollment ID: {0}, Course ID: {1}", 
               enrollment.EnrollmentID, enrollment.CourseID);
         }

         Console.ReadKey();
      }
   }
}

Lorsque l'exemple ci-dessus est exécuté, vous recevrez la sortie suivante. Tout d'abord, seules les informations sur l'étudiant sont affichées et après le chargement explicite de l'entité d'inscription, les informations sur l'étudiant et ses inscriptions associées sont affichées.

ID: 1, Name: Ali Alexander
Explicitly loaded Enrollments
ID: 1, Name: Ali Alexander
       Enrollment ID: 1, Course ID: 1050
       Enrollment ID: 2, Course ID: 4022
       Enrollment ID: 3, Course ID: 4041

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans ce chapitre, nous allons découvrir les techniques de validation qui peuvent être utilisées dans ADO.NET Entity Framework pour valider les données du modèle. Entity Framework fournit une grande variété de fonctionnalités de validation qui peuvent être implémentées sur une interface utilisateur pour la validation côté client ou peuvent être utilisées pour la validation côté serveur.

  • Dans Entity Framework, la validation des données fait partie de la solution pour intercepter les mauvaises données dans une application.

  • Entity Framework valide toutes les données avant qu'elles ne soient écrites dans la base de données par défaut, à l'aide d'un large éventail de méthodes de validation des données.

  • Cependant, Entity Framework vient après la validation des données de l'interface utilisateur. Donc, dans ce cas, il est nécessaire que la validation d'entité gère toutes les exceptions levées par EF et affiche un message générique.

  • Il existe certaines techniques de validation des données pour améliorer votre vérification des erreurs et comment renvoyer les messages d'erreur à l'utilisateur.

DbContext a une méthode Overridable appelée ValidateEntity. Lorsque vous appelez SaveChanges, Entity Framework appellera cette méthode pour chaque entité de son cache dont l'état n'est pas inchangé. Vous pouvez placer la logique de validation directement ici, comme illustré dans l'exemple suivant pour l'entité étudiante.

public partial class UniContextEntities : DbContext {

   protected override System.Data.Entity.Validation
      .DbEntityValidationResult ValidateEntity(DbEntityEntry entityEntry, 
      System.Collections.Generic.IDictionary<object, object> items) {

         if (entityEntry.Entity is Student) {

            if (entityEntry.CurrentValues.GetValue<string>("FirstMidName") == "") {

               var list = new List<System.Data.Entity
                  .Validation.DbValidationError>();

               list.Add(new System.Data.Entity.Validation
                  .DbValidationError("FirstMidName", "FirstMidName is required"));

               return new System.Data.Entity.Validation
                  .DbEntityValidationResult(entityEntry, list);
            }
         }

         if (entityEntry.CurrentValues.GetValue<string>("LastName") == "") {

            var list = new List<System.Data.Entity
               .Validation.DbValidationError>();

            list.Add(new System.Data.Entity.Validation
               .DbValidationError("LastName", "LastName is required"));

            return new System.Data.Entity.Validation
               .DbEntityValidationResult(entityEntry, list);
         }

         return base.ValidateEntity(entityEntry, items);
   }
}

Dans la méthode ValidateEntity ci-dessus, les propriétés FirstMidName et LastName de l'entité Student sont vérifiées si l'une de ces propriétés a une chaîne vide, puis elle renverra un message d'erreur.

Jetons un coup d'œil à un exemple simple dans lequel un nouvel étudiant est créé, mais le FirstMidName de l'étudiant est une chaîne vide, comme indiqué dans le code suivant.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         Console.WriteLine("Adding new Student to the database");
         Console.WriteLine();

         try {

            context.Students.Add(new Student() {
               FirstMidName = "",
               LastName = "Upston"
            });

            context.SaveChanges();
         } catch (DbEntityValidationException dbValidationEx) {

            foreach (DbEntityValidationResult entityErr in 
               dbValidationEx.EntityValidationErrors) {

               foreach (DbValidationError error in entityErr.ValidationErrors) {
                  Console.WriteLine("Error: {0}",error.ErrorMessage);
               }
            }
         }

         Console.ReadKey();
      }
   }
}

Lorsque l'exemple ci-dessus est compilé et exécuté, vous recevrez le message d'erreur suivant dans la fenêtre de la console.

Adding new Student to the database  
Error: FirstMidName is required

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Entity Framework permet de suivre les modifications apportées aux entités et à leurs relations, de sorte que les mises à jour correctes sont effectuées sur la base de données lorsque la méthode SaveChanges de contexte est appelée. Il s'agit d'une fonctionnalité clé d'Entity Framework.

  • Le suivi des modifications suit les modifications tout en ajoutant de nouveaux enregistrements à la collection d'entités, en modifiant ou en supprimant des entités existantes.

  • Ensuite, toutes les modifications sont conservées par le niveau DbContext.

  • Ces modifications de suivi sont perdues si elles ne sont pas enregistrées avant la destruction de l'objet DbContext.

  • La classe DbChangeTracker vous donne toutes les informations sur les entités actuelles suivies par le contexte.

  • Pour suivre une entité par le contexte, elle doit avoir la propriété de clé primaire.

Dans Entity Framework, le suivi des modifications est activé par défaut. Vous pouvez également désactiver le suivi des modifications en définissant la propriété AutoDetectChangesEnabled de DbContext sur false. Si cette propriété est définie sur true, Entity Framework conserve l'état des entités.

using (var context = new UniContextEntities()) {
   context.Configuration.AutoDetectChangesEnabled = true;
}

Jetons un coup d'œil à l'exemple suivant dans lequel les étudiants et leurs inscriptions sont extraits de la base de données.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         context.Configuration.AutoDetectChangesEnabled = true;
         Console.WriteLine("Retrieve Student");

         var student = (from s in context.Students where s.FirstMidName == 
            "Ali" select s).FirstOrDefault<Student>();

         string name = student.FirstMidName + " " + student.LastName;
         Console.WriteLine("ID: {0}, Name: {1}", student.ID, name);
         Console.WriteLine();
         Console.WriteLine("Retrieve all related enrollments");

         foreach (var enrollment in student.Enrollments) {
            Console.WriteLine("Enrollment ID: {0}, Course ID: {1}", 
               enrollment.EnrollmentID, enrollment.CourseID);
         }

         Console.WriteLine();

         Console.WriteLine("Context tracking changes of {0} entity.", 
            context.ChangeTracker.Entries().Count());

         var entries = context.ChangeTracker.Entries();

         foreach (var entry in entries) {
            Console.WriteLine("Entity Name: {0}", entry.Entity.GetType().Name);
            Console.WriteLine("Status: {0}", entry.State);
         }

         Console.ReadKey();
      }
   }
}

Lorsque l'exemple ci-dessus est compilé et exécuté, vous recevrez la sortie suivante.

Retrieve Student 
ID: 1, Name: Ali Alexander
Retrieve all related enrollments
       Enrollment ID: 1, Course ID: 1050
       Enrollment ID: 2, Course ID: 4022
       Enrollment ID: 3, Course ID: 4041
Context tracking changes of 4 entity.
Entity Name: Student
Status: Unchanged
Entity Name: Enrollment
Status: Unchanged
Entity Name: Enrollment
Status: Unchanged
Entity Name: Enrollment
Status: Unchanged

Vous pouvez voir que toutes les données sont uniquement extraites de la base de données, c'est pourquoi le statut est inchangé pour toutes les entités.

Jetons maintenant un coup d'œil à un autre exemple simple dans lequel nous ajouterons une inscription supplémentaire et supprimerons un étudiant de la base de données. Voici le code dans lequel une nouvelle inscription est ajoutée et un étudiant est supprimé.

class Program {

   static void Main(string[] args) {

      using (var context = new UniContextEntities()) {

         context.Configuration.AutoDetectChangesEnabled = true;

         Enrollment enr = new Enrollment() { 
            StudentID = 1, CourseID = 3141 
         };

         Console.WriteLine("Adding New Enrollment");
         context.Enrollments.Add(enr);
         Console.WriteLine("Delete Student");

         var student = (from s in context.Students where s.ID == 
            23 select s).SingleOrDefault<Student>();

         context.Students.Remove(student);
         Console.WriteLine("");

         Console.WriteLine("Context tracking changes of {0} entity.", 
            context.ChangeTracker.Entries().Count());
         var entries = context.ChangeTracker.Entries();

         foreach (var entry in entries) {
            Console.WriteLine("Entity Name: {0}", entry.Entity.GetType().Name);
            Console.WriteLine("Status: {0}", entry.State);
         }

         Console.ReadKey();
      }
   }
}

Lorsque l'exemple ci-dessus est compilé et exécuté, vous recevrez la sortie suivante.

Adding New Enrollment
Delete Student
Context tracking changes of 2 entity.
Entity Name: Enrollment
Status: Added
Entity Name: Student
Status: Deleted

Vous pouvez maintenant voir que le statut de l'entité d'inscription est défini sur ajouté et que le statut de l'entité étudiant est supprimé, car une nouvelle inscription a été ajoutée et un étudiant est supprimé de la base de données.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans Entity Framework, Colored Entity consiste principalement à changer la couleur de l'entité dans le concepteur afin qu'il soit facile pour les développeurs d'identifier les groupes d'entités associés dans le concepteur Visual Studio. Cette fonctionnalité a été introduite pour la première fois dans Entity Framework 5.0.

  • Cette fonctionnalité n'a rien à voir avec les aspects de performances.

  • Lorsque vous avez un projet à grande échelle et de nombreuses entités dans un fichier edmx, cette fonctionnalité est très utile pour séparer vos entités dans différents modules.

Si vous travaillez avec un fichier edmx et que vous l'avez ouvert dans le concepteur, pour changer la couleur, sélectionnez une entité dans les fenêtres de conception. Cliquez ensuite avec le bouton droit de la souris et sélectionnez Propriétés.

Dans la fenêtre Propriétés, sélectionnez la propriété Couleur de remplissage.

Spécifiez la couleur en utilisant un nom de couleur valide, par exemple, Vert ou un RVB valide (255, 128, 128) ou vous pouvez également sélectionner dans le sélecteur de couleur.

Pour changer la couleur de plusieurs entités en une seule fois, sélectionnez plusieurs entités et modifiez la couleur de remplissage pour toutes à l'aide de la fenêtre de propriétés.

Vous pouvez également modifier le format des propriétés en sélectionnant l'une des options suivantes -

  • Afficher un nom
  • Nom et type d'affichage

Par défaut, l'option de nom d'affichage est sélectionnée. Pour modifier le format de la propriété, cliquez avec le bouton droit de la souris sur la fenêtre du concepteur.

Sélectionnez Format de propriété scalaire → Afficher le nom et le type.

Vous pouvez maintenant voir que le type est également affiché avec le nom.

L'Entity Framework propose trois approches pour créer un modèle d'entité et chacune a ses propres avantages et inconvénients.

  • Code d'abord
  • La base de données d'abord
  • Modèle d'abord

Dans ce chapitre, nous décrirons brièvement la première approche du code. Certains développeurs préfèrent travailler avec le Designer dans Code tandis que d'autres préfèrent simplement travailler avec leur code. Pour ces développeurs, Entity Framework dispose d'un workflow de modélisation appelé Code First.

  • Le workflow de modélisation Code First cible une base de données qui n'existe pas et Code First la créera.

  • Il peut également être utilisé si vous avez une base de données vide et que Code First y ajoutera de nouvelles tables.

  • Code First vous permet de définir votre modèle à l'aide des classes C # ou VB.Net.

  • Une configuration supplémentaire peut éventuellement être effectuée à l'aide d'attributs sur vos classes et propriétés ou à l'aide d'une API fluide.

Pourquoi Code First?

  • Code First est vraiment composé d'un ensemble de pièces de puzzle. Premièrement, vos classes de domaine.

  • Les classes de domaine n'ont rien à voir avec Entity Framework. Ce ne sont que les éléments de votre domaine professionnel.

  • Entity Framework a donc un contexte qui gère l'interaction entre ces classes et votre base de données.

  • Le contexte n'est pas spécifique à Code First. C'est une fonctionnalité Entity Framework.

  • Code First ajoute un générateur de modèle qui inspecte vos classes gérées par le contexte, puis utilise un ensemble de règles ou de conventions pour déterminer comment ces classes et les relations décrivent un modèle et comment ce modèle doit être mappé à votre base de données.

  • Tout cela se produit au moment de l'exécution. Vous ne verrez jamais ce modèle, c'est juste en mémoire.

  • Code First a également la possibilité d'utiliser ce modèle pour créer une base de données si vous le souhaitez.

  • Il peut également mettre à jour la base de données si le modèle change, à l'aide d'une fonctionnalité appelée Migrations Code First.

Configuration de l'environnement

Pour commencer à utiliser l'approche EF Code First, vous devez installer les outils suivants sur votre système.

  • Visual Studio 2013 (.net framework 4.5.2) ou version ultérieure.
  • MS SQL Server 2012 ou plus récent.
  • Entity Framework via NuGet Package.

Installer EF via le package NuGet

Step 1 - Tout d'abord, créez l'application console depuis Fichier → Nouveau → Projet…

Step 2 - Sélectionnez Windows dans le volet gauche et Application console dans le volet modèle.

Step 3 - Entrez EFCodeFirstDemo comme nom et sélectionnez OK.

Step 4 - Faites un clic droit sur votre projet dans l'explorateur de solutions et sélectionnez Gérer les packages NuGet…

Cela ouvrira NuGet Package Manager et recherchera EntityFramework. Cela recherchera tous les packages liés à Entity Framework.

Step 5- Sélectionnez EntityFramework et cliquez sur Installer. Ou dans le menu Outils, cliquez sur Gestionnaire de package NuGet, puis sur Console du gestionnaire de package. Dans la fenêtre de la console du gestionnaire de package, entrez la commande suivante: Install-Package EntityFramework.

Lorsque l'installation est terminée, vous verrez le message suivant dans la fenêtre de sortie «Installation réussie de« EntityFramework 6.1.2 »sur EFCodeFirstDemo».

Après l'installation, EntityFramework.dll sera inclus dans votre projet, comme illustré dans l'image suivante.

Vous êtes maintenant prêt à commencer à travailler sur l'approche Code First.

Définissons un modèle très simple en utilisant des classes. Nous les définissons simplement dans le fichier Program.cs, mais dans une application du monde réel, vous diviserez vos classes en fichiers séparés et potentiellement en un projet distinct. Voici un modèle de données que nous allons créer en utilisant l'approche Code First.

Créer un modèle

Ajoutez les trois classes suivantes dans le fichier Program.cs à l'aide du code suivant pour la classe Student.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}
  • La propriété ID deviendra la colonne de clé primaire de la table de base de données correspondant à cette classe.

  • La propriété Enrollments est une propriété de navigation. Les propriétés de navigation contiennent d'autres entités liées à cette entité.

  • Dans ce cas, la propriété Enrollments d'une entité Student contiendra toutes les entités Enrollment qui sont liées à cette entité Student.

  • Les propriétés de navigation sont généralement définies comme virtuelles afin de pouvoir tirer parti de certaines fonctionnalités d'Entity Framework telles que le chargement différé.

  • Si une propriété de navigation peut contenir plusieurs entités (comme dans les relations plusieurs-à-plusieurs ou un-à-plusieurs), son type doit être une liste dans laquelle des entrées peuvent être ajoutées, supprimées et mises à jour, telles que ICollection.

Voici l'implémentation de la classe Course.

public class Course {
   public int CourseID { get; set; }
   public string Title { get; set; }
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

La propriété Enrollments est une propriété de navigation. Une entité de cours peut être liée à un nombre quelconque d'entités d'inscription.

Voici l'implémentation de la classe d'inscription et de l'énumération.

public enum Grade {
   A, B, C, D, F
}

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }
}
  • La propriété EnrollmentID sera la clé primaire.

  • La propriété Grade est une énumération. Le point d'interrogation après la déclaration de type Grade indique que la propriété Grade est nullable.

  • Une note nulle est différente d'une note zéro. Null signifie qu'une note n'est pas connue ou n'a pas encore été attribuée.

  • Les propriétés StudentID et CourseID sont des clés étrangères et les propriétés de navigation correspondantes sont Student et Course.

  • Une entité Inscription est associée à une entité Etudiant et à une entité Cours, de sorte que la propriété ne peut contenir qu'une seule entité Etudiant et Cours.

Créer un contexte de base de données

La classe principale qui coordonne la fonctionnalité Entity Framework pour un modèle de données donné est la classe de contexte de base de données qui permet d'interroger et d'enregistrer des données. Vous pouvez créer cette classe en dérivant de la classe DbContext et en exposant un DbSet typé pour chaque classe de notre modèle. Voici l'implémentation sur la classe MyContext, qui est dérivée de la classe DbContext.

public class MyContext : DbContext {
   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

Voici le code complet dans le fichier Program.cs.

using System.ComponentModel.DataAnnotations.Schema;
using System.Data.Entity;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace EFCodeFirstDemo {

   class Program {
      static void Main(string[] args) {}
   }

   public enum Grade {
      A, B, C, D, F
   }

   public class Enrollment {
      public int EnrollmentID { get; set; }
      public int CourseID { get; set; }
      public int StudentID { get; set; }
      public Grade? Grade { get; set; }
		
      public virtual Course Course { get; set; }
      public virtual Student Student { get; set; }
   }

   public class Student {
      public int ID { get; set; }
      public string LastName { get; set; }
      public string FirstMidName { get; set; }
      public DateTime EnrollmentDate { get; set; }
		
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }

   public class Course {
      public int CourseID { get; set; }
      public string Title { get; set; }
      public int Credits { get; set; }
		
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }

   public class MyContext : DbContext {
      public virtual DbSet<Course> Courses { get; set; }
      public virtual DbSet<Enrollment> Enrollments { get; set; }
      public virtual DbSet<Student> Students { get; set; }
   }

}

Le code ci-dessus est tout ce dont nous avons besoin pour commencer à stocker et à récupérer des données. Ajoutons quelques données, puis récupérons-les. Voici le code de la méthode principale.

static void Main(string[] args) {

   using (var context = new MyContext()) {
      // Create and save a new Students
      Console.WriteLine("Adding new students");

      var student = new Student {
         FirstMidName = "Alain", LastName = "Bomer", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
      };

      context.Students.Add(student);
		
      var student1 = new Student {
         FirstMidName = "Mark", LastName = "Upston", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
      };

      context.Students.Add(student1);
      context.SaveChanges();

      // Display all Students from the database
      var students = (from s in context.Students 
         orderby s.FirstMidName select s).ToList<Student>();

      Console.WriteLine("Retrieve all Students from the database:");

      foreach (var stdnt in students) {
         string name = stdnt.FirstMidName + " " + stdnt.LastName;
         Console.WriteLine("ID: {0}, Name: {1}", stdnt.ID, name);
      }
		
      Console.WriteLine("Press any key to exit...");
      Console.ReadKey();
   }
}

Lorsque le code ci-dessus est exécuté, vous recevrez la sortie suivante.

Adding new students
Retrieve all Students from the database:
ID: 1, Name: Alain Bomer
ID: 2, Name: Mark Upston
Press any key to exit...

Maintenant, la question qui me vient à l'esprit est de savoir où se trouvent les données et la base de données dans lesquelles nous avons ajouté des données, puis les avons extraites de la base de données. Par convention, DbContext a créé une base de données pour vous.

  • Si une instance SQL Express locale est disponible, Code First a créé la base de données sur cette instance.

  • Si SQL Express n'est pas disponible, Code First essaiera d'utiliser LocalDb.

  • La base de données est nommée d'après le nom qualifié complet du contexte dérivé.

Dans notre cas, l'instance SQL Express est disponible et le nom de la base de données est EFCodeFirstDemo.MyContext, comme illustré dans l'image suivante.

  • Ce ne sont que les conventions par défaut et il existe différentes manières de modifier la base de données utilisée par Code First.

  • Comme vous pouvez le voir dans l'image ci-dessus, il a créé des tables Etudiants, Cours et Inscriptions et chaque table contient des colonnes avec le type de données et la longueur appropriés.

  • Les noms de colonne et le type de données correspondent également aux propriétés des classes de domaine respectives.

Initialisation de la base de données

Dans l'exemple ci-dessus, nous avons vu que Code First crée automatiquement une base de données, mais si vous souhaitez changer le nom de la base de données et du serveur, voyons comment Code First décide du nom et du serveur de la base de données lors de l'initialisation d'une base de données. Jetez un œil au diagramme suivant.

Vous pouvez définir le constructeur de base de la classe de contexte des manières suivantes.

  • Aucun paramètre
  • Nom de la base de données
  • Nom de la chaîne de connexion

Aucun paramètre

Si vous spécifiez le constructeur de base de la classe de contexte sans aucun paramètre comme indiqué dans l'exemple ci-dessus, alors le framework d'entité créera une base de données dans votre serveur SQLEXPRESS local avec un nom {Namespace}. {Context class name}.

Dans l'exemple ci-dessus, la base de données créée automatiquement porte le nom EFCodeFirstDemo.MyContext. Si vous regardez le nom, vous constaterez que EFCodeFirstDemo est l'espace de noms et MyContext est le nom de la classe de contexte, comme indiqué dans le code suivant.

public class MyContext : DbContext {
   public MyContext() : base() {}

   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

Nom de la base de données

Si vous passez le nom de la base de données en tant que paramètre dans un constructeur de base de la classe de contexte, alors Code First créera à nouveau une base de données automatiquement, mais cette fois le nom sera celui passé en paramètre dans le constructeur de base sur le serveur de base de données local SQLEXPRESS .

Dans le code suivant, MyContextDB est spécifié en tant que paramètre dans le constructeur de base. Si vous exécutez votre application, la base de données portant le nom MyContextDB sera créée sur votre serveur SQL local.

public class MyContext : DbContext {
   public MyContext() : base("MyContextDB") {}
   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

Nom de la chaîne de connexion

Il s'agit d'un moyen simple d'indiquer à DbContext d'utiliser un serveur de base de données autre que SQL Express ou LocalDb. Vous pouvez choisir de mettre une chaîne de connexion dans votre fichier app.config.

  • Si le nom de la chaîne de connexion correspond au nom de votre contexte (avec ou sans qualification d'espace de noms), alors il sera trouvé par DbContext lorsque le paramètre moins constructeur est utilisé.

  • Si le nom de la chaîne de connexion est différent du nom de votre contexte, vous pouvez indiquer à DbContext d'utiliser cette connexion en mode Code First en transmettant le nom de la chaîne de connexion au constructeur DbContext.

public class MyContext : DbContext {
   public MyContext() : base("name = MyContextDB") {}
   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}
  • Dans le code ci-dessus, un extrait de chaîne de connexion de classe de contexte est spécifié en tant que paramètre dans le constructeur de base.

  • Le nom de la chaîne de connexion doit commencer par "name =" sinon, il le considérera comme un nom de base de données.

  • Ce formulaire indique explicitement que vous vous attendez à ce que la chaîne de connexion soit trouvée dans votre fichier de configuration. Une exception sera levée si une chaîne de connexion avec le nom donné n'est pas trouvée.

<connectionStrings>
   <add name = "MyContextDB"
      connectionString = "Data Source =.;Initial Catalog = EFMyContextDB;Integrated Security = true"
      providerName = "System.Data.SqlClient"/>
</connectionStrings>
  • Le nom de la base de données dans la chaîne de connexion dans app.config est EFMyContextDB. CodeFirst créera un nouveauEFMyContextDB base de données ou utiliser existant EFMyContextDB base de données sur SQL Server local.

Classes de domaine

Jusqu'à présent, nous avons simplement laissé EF découvrir le modèle en utilisant ses conventions par défaut, mais il y aura des moments où nos classes ne suivent pas les conventions et nous devons être en mesure d'effectuer une configuration supplémentaire. Mais vous pouvez remplacer ces conventions en configurant vos classes de domaine pour fournir à EF les informations dont il a besoin. Il existe deux options pour configurer vos classes de domaine -

  • Annotations de données
  • API Fluent

Annotations de données

DataAnnotations est utilisé pour configurer vos classes qui mettront en évidence les configurations les plus couramment nécessaires. Les DataAnnotations sont également comprises par un certain nombre d'applications .NET, telles que ASP.NET MVC, qui permettent à ces applications d'exploiter les mêmes annotations pour les validations côté client.

Voici les annotations de données utilisées en classe d'élève.

public class Enrollment {

   [Key]
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }

   [ForeignKey("CourseID")]
   public virtual Course Course { get; set; }

   [ForeignKey("ID")]
   public virtual Student Student { get; set; }
}

API Fluent

La plupart des configurations de modèle peuvent être effectuées à l'aide d'annotations de données simples. L'API Fluent est un moyen avancé de spécifier la configuration du modèle qui couvre tout ce que les annotations de données peuvent faire, en plus d'une configuration plus avancée impossible avec les annotations de données. Les annotations de données et l'API fluent peuvent être utilisées ensemble.

Pour accéder à l'API fluent, vous substituez la méthode OnModelCreating dans DbContext. Renommons maintenant le nom de la colonne dans la table des étudiants de FirstMidName en FirstName, comme indiqué dans le code suivant.

public class MyContext : DbContext {

   protected override void OnModelCreating(DbModelBuilder modelBuilder) {
      modelBuilder.Entity<Student>().Property(s ⇒ s.FirstMidName)
         .HasColumnName("FirstName");
   }

   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

DataAnnotations est utilisé pour configurer les classes qui mettront en évidence les configurations les plus couramment nécessaires. Les annotations de données sont également comprises par un certain nombre d'applications .NET, telles que ASP.NET MVC qui permet à ces applications d'exploiter les mêmes annotations pour les validations côté client. Les attributs DataAnnotation remplacent les conventions CodeFirst par défaut.

System.ComponentModel.DataAnnotations inclut les attributs suivants qui ont un impact sur la nullité ou la taille de la colonne.

  • Key
  • Timestamp
  • ConcurrencyCheck
  • Required
  • MinLength
  • MaxLength
  • StringLength

System.ComponentModel.DataAnnotations.Schema namespace inclut les attributs suivants qui ont un impact sur le schéma de la base de données.

  • Table
  • Column
  • Index
  • ForeignKey
  • NotMapped
  • InverseProperty

Clé

Entity Framework repose sur chaque entité ayant une valeur de clé qu'elle utilise pour suivre les entités. L'une des conventions dont dépend Code First est de savoir comment elle implique quelle propriété est la clé dans chacune des classes Code First.

  • La convention consiste à rechercher une propriété nommée «Id» ou une propriété qui combine le nom de la classe et «Id», comme «StudentId».

  • La propriété correspondra à une colonne de clé primaire dans la base de données.

  • Les classes d'étudiant, de cours et d'inscription suivent cette convention.

Supposons maintenant que la classe Student utilise le nom StdntID au lieu de ID. Lorsque Code First ne trouve pas de propriété qui correspond à cette convention, il lèvera une exception en raison de l'exigence d'Entity Framework selon laquelle vous devez avoir une propriété de clé. Vous pouvez utiliser l'annotation de clé pour spécifier la propriété à utiliser comme EntityKey.

Jetons un coup d'œil au code suivant d'une classe Student qui contient StdntID, mais il ne suit pas la convention Code First par défaut. Donc, pour gérer cela, un attribut Key est ajouté qui en fera une clé primaire.

public class Student {

   [Key]
   public int StdntID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Lorsque vous exécutez votre application et examinez votre base de données dans SQL Server Explorer, vous verrez que la clé primaire est désormais StdntID dans la table des étudiants.

Entity Framework prend également en charge les clés composites. Composite keyssont également des clés primaires constituées de plusieurs propriétés. Par exemple, vous avez une classe DrivingLicense dont la clé primaire est une combinaison de LicenseNumber et IssuingCountry.

public class DrivingLicense {

   [Key, Column(Order = 1)]
   public int LicenseNumber { get; set; }
   [Key, Column(Order = 2)]
   public string IssuingCountry { get; set; }
   public DateTime Issued { get; set; }
   public DateTime Expires { get; set; }
}

Lorsque vous avez des clés composites, Entity Framework vous oblige à définir un ordre des propriétés de clé. Vous pouvez le faire en utilisant l'annotation Colonne pour spécifier une commande.

Horodatage

Code First traitera les propriétés d'horodatage de la même manière que les propriétés ConcurrencyCheck, mais il garantira également que le champ de base de données que le code génère en premier ne peut pas être nul.

  • Il est plus courant d'utiliser des champs rowversion ou timestamp pour la vérification de la concurrence.

  • Plutôt que d'utiliser l'annotation ConcurrencyCheck, vous pouvez utiliser l'annotation TimeStamp plus spécifique tant que le type de la propriété est un tableau d'octets.

  • Vous ne pouvez avoir qu'une seule propriété d'horodatage dans une classe donnée.

Jetons un coup d'œil à un exemple simple en ajoutant la propriété TimeStamp à la classe Course -

public class Course {

   public int CourseID { get; set; }
   public string Title { get; set; }
   public int Credits { get; set; }
   [Timestamp]
   public byte[] TStamp { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Comme vous pouvez le voir dans l'exemple ci-dessus, l'attribut Timestamp est appliqué à la propriété Byte [] de la classe Course. Ainsi, Code First créera une colonne d'horodatage TStampdans la table Courses.

Vérification de la concurrence

L'annotation ConcurrencyCheck vous permet de marquer une ou plusieurs propriétés à utiliser pour la vérification d'accès concurrentiel dans la base de données lorsqu'un utilisateur modifie ou supprime une entité. Si vous avez travaillé avec EF Designer, cela correspond à la définition de ConcurrencyMode d'une propriété sur Fixed.

Jetons un coup d'œil à un exemple simple du fonctionnement de ConcurrencyCheck en l'ajoutant à la propriété Title dans la classe Course.

public class Course {

   public int CourseID { get; set; }
   [ConcurrencyCheck]
   public string Title { get; set; }
   public int Credits { get; set; }
   [Timestamp, DataType("timestamp")]
   public byte[] TimeStamp { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Dans la classe Course ci-dessus, l'attribut ConcurrencyCheck est appliqué à la propriété Title existante. Désormais, Code First inclura la colonne Titre dans la commande de mise à jour pour vérifier la concurrence optimiste, comme indiqué dans le code suivant.

exec sp_executesql N'UPDATE [dbo].[Courses]
   SET [Title] = @0
   WHERE (([CourseID] = @1) AND ([Title] = @2))
   ',N'@0 nvarchar(max) ,@1 int,@2 nvarchar(max) ',@0=N'Maths',@1=1,@2=N'Calculus'
go

Annotation requise

L'annotation requise indique à EF qu'une propriété particulière est requise. Jetons un coup d'œil à la classe Student suivante dans laquelle l'ID requis est ajouté à la propriété FirstMidName. L'attribut obligatoire forcera EF à s'assurer que la propriété contient des données.

public class Student {

   [Key]
   public int StdntID { get; set; }

   [Required]
   public string LastName { get; set; }

   [Required]
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Comme vu dans l'exemple ci-dessus, l'attribut requis est appliqué à FirstMidName et LastName. Ainsi, Code First créera des colonnes NOT NULL FirstMidName et LastName dans la table Students, comme illustré dans l'image suivante.

Longueur maximale

L'attribut MaxLength vous permet de spécifier des validations de propriété supplémentaires. Il peut être appliqué à une propriété de type chaîne ou tableau d'une classe de domaine. EF Code First définira la taille d'une colonne comme spécifié dans l'attribut MaxLength.

Jetons un coup d'œil à la classe Course suivante dans laquelle l'attribut MaxLength (24) est appliqué à la propriété Title.

public class Course {

   public int CourseID { get; set; }
   [ConcurrencyCheck]
   [MaxLength(24)]
   public string Title { get; set; }
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Lorsque vous exécutez l'application ci-dessus, Code First crée un titre de colonne nvarchar (24) dans la table CourseId, comme illustré dans l'image suivante.

Lorsque l'utilisateur définit le titre qui contient plus de 24 caractères, EF lèvera EntityValidationError.

Longueur minimale

L'attribut MinLength vous permet également de spécifier des validations de propriétés supplémentaires, comme vous l'avez fait avec MaxLength. L'attribut MinLength peut également être utilisé avec l'attribut MaxLength comme indiqué dans le code suivant.

public class Course {

   public int CourseID { get; set; }
   [ConcurrencyCheck]
   [MaxLength(24) , MinLength(5)]
   public string Title { get; set; }
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

EF lèvera EntityValidationError, si vous définissez une valeur de propriété Title inférieure à la longueur spécifiée dans l'attribut MinLength ou supérieure à la longueur spécifiée dans l'attribut MaxLength.

Longueur de chaine

StringLength vous permet également de spécifier des validations de propriétés supplémentaires telles que MaxLength. La seule différence est que l'attribut StringLength ne peut être appliqué qu'à une propriété de type chaîne de classes Domain.

public class Course {

   public int CourseID { get; set; }
   [StringLength (24)]
   public string Title { get; set; }
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Entity Framework valide également la valeur d'une propriété pour l'attribut StringLength. Si l'utilisateur définit le titre qui contient plus de 24 caractères, EF lèvera EntityValidationError.

Table

La convention Code par défaut First crée un nom de table similaire au nom de classe. Si vous laissez Code First créer la base de données et souhaitez également modifier le nom des tables qu'il crée. Puis -

  • Vous pouvez utiliser Code First avec une base de données existante. Mais ce n'est pas toujours le cas que les noms des classes correspondent aux noms des tables de votre base de données.

  • L'attribut de table remplace cette convention par défaut.

  • EF Code First créera une table avec un nom spécifié dans l'attribut Table pour une classe de domaine donnée.

Jetons un coup d'œil à l'exemple suivant dans lequel la classe est nommée Student et, par convention, Code First suppose que cela correspondra à une table nommée Students. Si ce n'est pas le cas, vous pouvez spécifier le nom de la table avec l'attribut Table comme indiqué dans le code suivant.

[Table("StudentsInfo")]
public class Student {

   [Key]
   public int StdntID { get; set; }
   [Required]
   public string LastName { get; set; }
   [Required]
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez maintenant voir que l'attribut Table spécifie la table en tant que StudentsInfo. Lorsque la table est générée, vous verrez le nom de table StudentsInfo comme indiqué dans l'image suivante.

Vous pouvez non seulement spécifier le nom de la table, mais vous pouvez également spécifier un schéma pour la table à l'aide de l'attribut Table comme indiqué dans le code suivant.

[Table("StudentsInfo", Schema = "Admin")] 
public class Student {

   [Key]
   public int StdntID { get; set; }
   [Required]
   public string LastName { get; set; }
   [Required]
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez voir dans l'exemple ci-dessus, la table est spécifiée avec le schéma d'administration. Maintenant, Code First va créer la table StudentsInfo dans le schéma Admin, comme indiqué dans l'image suivante.

Colonne

Il est également identique à l'attribut Table, mais l'attribut Table remplace le comportement de la table tandis que l'attribut Column remplace le comportement de la colonne. La convention Code par défaut First crée un nom de colonne similaire au nom de propriété. Si vous laissez Code First créer la base de données et souhaitez également modifier le nom des colonnes de vos tables. Puis -

  • L'attribut de colonne remplace la convention par défaut.

  • EF Code First créera une colonne avec un nom spécifié dans l'attribut Column pour une propriété donnée.

Jetons un coup d'œil à l'exemple suivant dans lequel la propriété est nommée FirstMidName et par convention, Code First suppose que cela correspondra à une colonne nommée FirstMidName.

Si ce n'est pas le cas, vous pouvez spécifier le nom de la colonne avec l'attribut Column comme indiqué dans le code suivant.

public class Student {

   public int ID { get; set; }
   public string LastName { get; set; }
   [Column("FirstName")]
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez voir que l'attribut Column spécifie la colonne comme FirstName. Lorsque la table est générée, vous verrez le nom de colonne FirstName comme indiqué dans l'image suivante.

Indice

L'attribut Index a été introduit dans Entity Framework 6.1. Si vous utilisez une version antérieure, les informations de cette section ne s'appliquent pas.

  • Vous pouvez créer un index sur une ou plusieurs colonnes à l'aide de IndexAttribute.

  • L'ajout de l'attribut à une ou plusieurs propriétés obligera EF à créer l'index correspondant dans la base de données lors de la création de la base de données.

  • Les index rendent la récupération des données plus rapide et plus efficace, dans la plupart des cas. Cependant, la surcharge d'une table ou d'une vue avec des index peut affecter de manière désagréable les performances d'autres opérations telles que les insertions ou les mises à jour.

  • L'indexation est la nouvelle fonctionnalité d'Entity Framework qui vous permet d'améliorer les performances de votre application Code First en réduisant le temps requis pour interroger les données de la base de données.

  • Vous pouvez ajouter des index à votre base de données à l'aide de l'attribut Index et remplacer les paramètres Unique et Clustered par défaut pour obtenir l'index le mieux adapté à votre scénario.

  • Par défaut, l'index sera nommé IX_ <nom de la propriété>

Jetons un coup d'œil au code suivant dans lequel l'attribut Index est ajouté dans la classe Course pour les crédits.

public class Course {
   public int CourseID { get; set; }
   public string Title { get; set; }
   [Index]
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez voir que l'attribut Index est appliqué à la propriété Credits. Lorsque la table est générée, vous verrez IX_Credits dans les index.

Par défaut, les index ne sont pas uniques, mais vous pouvez utiliser le IsUniqueparamètre nommé pour spécifier qu'un index doit être unique. L'exemple suivant présente un index unique comme indiqué dans le code suivant.

public class Course {
   public int CourseID { get; set; }
   [Index(IsUnique = true)]
	
   public string Title { get; set; }
   [Index]
	
   public int Credits { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Clé étrangère

La convention Code First s'occupe des relations les plus courantes dans votre modèle, mais dans certains cas, elle a besoin d'aide. Par exemple, la modification du nom de la propriété clé dans la classe Student a créé un problème avec sa relation avec la classe Enrollment.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }
}

public class Student {
   [Key]
   public int StdntID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Lors de la génération de la base de données, Code First voit la propriété StudentID dans la classe Enrollment et la reconnaît, par la convention qu'elle correspond à un nom de classe plus «ID», comme une clé étrangère de la classe Student. Cependant, il n'existe aucune propriété StudentID dans la classe Student, mais il s'agit de la propriété StdntID de la classe Student.

La solution pour cela consiste à créer une propriété de navigation dans l'inscription et à utiliser le ForeignKey DataAnnotation pour aider Code First à comprendre comment créer la relation entre les deux classes, comme indiqué dans le code suivant.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
	
   public Grade? Grade { get; set; }
   public virtual Course Course { get; set; }
   [ForeignKey("StudentID")]
	
   public virtual Student Student { get; set; }
}

Vous pouvez voir maintenant que l'attribut ForeignKey est appliqué à la propriété de navigation.

Non mappé

Par défaut, les conventions de Code First, chaque propriété qui est d'un type de données pris en charge et qui inclut des getters et des setters est représentée dans la base de données. Mais ce n'est pas toujours le cas dans vos applications. L'attribut NotMapped remplace cette convention par défaut. Par exemple, vous pouvez avoir une propriété dans la classe Student telle que FatherName, mais elle n'a pas besoin d'être stockée. Vous pouvez appliquer l'attribut NotMapped à une propriété FatherName dont vous ne souhaitez pas créer de colonne dans la base de données, comme indiqué dans le code suivant.

public class Student {
   [Key]
   public int StdntID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
	
   public DateTime EnrollmentDate { get; set; }
   [NotMapped]

   public int FatherName { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Vous pouvez voir que l'attribut NotMapped est appliqué à la propriété FatherName. Lorsque la table est générée, vous verrez que la colonne FatherName ne sera pas créée dans une base de données, mais elle est présente dans la classe Student.

Code First ne crée pas de colonne pour une propriété, qui n'a ni getters ni setters, comme illustré dans l'exemple suivant des propriétés Address et Age de la classe Student.

Propriété inverse

InverseProperty est utilisé lorsque vous avez plusieurs relations entre les classes. Dans la classe d'inscription, vous souhaiterez peut-être garder une trace de qui s'est inscrit au cours actuel et au cours précédent. Ajoutons deux propriétés de navigation pour la classe Enrollment.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course CurrCourse { get; set; }
   public virtual Course PrevCourse { get; set; }
   public virtual Student Student { get; set; }
}

De même, vous devrez également ajouter la classe Course référencée par ces propriétés. La classe Course a des propriétés de navigation vers la classe Inscription, qui contient toutes les inscriptions actuelles et précédentes.

public class Course {

   public int CourseID { get; set; }
   public string Title { get; set; }
   [Index]

   public int Credits { get; set; }
   public virtual ICollection<Enrollment> CurrEnrollments { get; set; }
   public virtual ICollection<Enrollment> PrevEnrollments { get; set; }
}

Code First crée la colonne de clé étrangère {Nom de classe} _ {Clé primaire}, si la propriété de clé étrangère n'est pas incluse dans une classe particulière, comme indiqué dans les classes ci-dessus. Lorsque la base de données est générée, vous verrez les clés étrangères suivantes.

Comme vous pouvez le voir, Code first n'est pas en mesure de faire correspondre les propriétés des deux classes par lui-même. La table de base de données pour les inscriptions doit avoir une clé étrangère pour le CurrCourse et une pour le PrevCourse, mais Code First créera quatre propriétés de clé étrangère, c'est-à-dire

  • CurrCourse _CourseID
  • PrevCourse _CourseID
  • Course_CourseID et
  • Course_CourseID1

Pour résoudre ces problèmes, vous pouvez utiliser l'annotation InverseProperty pour spécifier l'alignement des propriétés.

public class Course {

   public int CourseID { get; set; }
   public string Title { get; set; }
   [Index]

   public int Credits { get; set; }
   [InverseProperty("CurrCourse")]

   public virtual ICollection<Enrollment> CurrEnrollments { get; set; }
   [InverseProperty("PrevCourse")]

   public virtual ICollection<Enrollment> PrevEnrollments { get; set; }
}

Comme vous pouvez le voir, l'attribut InverseProperty est appliqué dans la classe Course ci-dessus en spécifiant la propriété de référence de la classe d'inscription à laquelle il appartient. Désormais, Code First générera une base de données et créera uniquement deux colonnes de clé étrangère dans la table Enrollments, comme illustré dans l'image suivante.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

L'API Fluent est un moyen avancé de spécifier la configuration du modèle qui couvre tout ce que les annotations de données peuvent faire en plus d'une configuration plus avancée impossible avec les annotations de données. Les annotations de données et l'API fluent peuvent être utilisées ensemble, mais Code First donne la priorité à l'API Fluent> annotations de données> conventions par défaut.

  • L'API Fluent est un autre moyen de configurer vos classes de domaine.

  • L'API Code First Fluent est le plus souvent accessible en remplaçant la méthode OnModelCreating sur votre DbContext dérivé.

  • L'API Fluent fournit plus de fonctionnalités pour la configuration que les DataAnnotations. L'API Fluent prend en charge les types de mappages suivants.

Dans ce chapitre, nous continuerons avec l'exemple simple qui contient les classes Student, Course et Enrollment et une classe de contexte avec le nom MyContext comme indiqué dans le code suivant.

using System.Data.Entity; 
using System.Linq; 
using System.Text;
using System.Threading.Tasks;  

namespace EFCodeFirstDemo {

   class Program {
      static void Main(string[] args) {}
   }
   
   public enum Grade {
      A, B, C, D, F
   }

   public class Enrollment {
      public int EnrollmentID { get; set; }
      public int CourseID { get; set; }
      public int StudentID { get; set; }
      public Grade? Grade { get; set; }
		
      public virtual Course Course { get; set; }
      public virtual Student Student { get; set; }
   }

   public class Student {
      public int ID { get; set; }
      public string LastName { get; set; }
      public string FirstMidName { get; set; }
		
      public DateTime EnrollmentDate { get; set; }
		
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }

   public class Course {
      public int CourseID { get; set; }
      public string Title { get; set; }
      public int Credits { get; set; }
		
      public virtual ICollection<Enrollment> Enrollments { get; set; }
   }

   public class MyContext : DbContext {
      public virtual DbSet<Course> Courses { get; set; }
      public virtual DbSet<Enrollment> Enrollments { get; set; }
      public virtual DbSet<Student> Students { get; set; }
   }

}

Pour accéder à l'API Fluent, vous devez remplacer la méthode OnModelCreating dans DbContext. Jetons un coup d'œil à un exemple simple dans lequel nous renommerons le nom de la colonne dans la table des étudiants de FirstMidName en FirstName, comme indiqué dans le code suivant.

public class MyContext : DbContext {

   protected override void OnModelCreating(DbModelBuilder modelBuilder) {
      modelBuilder.Entity<Student>().Property(s ⇒ s.FirstMidName)
      .HasColumnName("FirstName");}

      public virtual DbSet<Course> Courses { get; set; }
      public virtual DbSet<Enrollment> Enrollments { get; set; }
      public virtual DbSet<Student> Students { get; set; }
}

DbModelBuilder est utilisé pour mapper les classes CLR à un schéma de base de données. C'est la classe principale et sur laquelle vous pouvez configurer toutes vos classes de domaine. Cette approche centrée sur le code pour créer un modèle de données d'entité (EDM) est connue sous le nom de Code First.

L'API Fluent fournit un certain nombre de méthodes importantes pour configurer les entités et ses propriétés afin de remplacer diverses conventions Code First. Voici quelques-uns d'entre eux.

Sr. No. Nom et description de la méthode
1

ComplexType<TComplexType>

Enregistre un type en tant que type complexe dans le modèle et renvoie un objet pouvant être utilisé pour configurer le type complexe. Cette méthode peut être appelée plusieurs fois pour le même type pour effectuer plusieurs lignes de configuration.

2

Entity<TEntityType>

Enregistre un type d'entité dans le cadre du modèle et renvoie un objet pouvant être utilisé pour configurer l'entité. Cette méthode peut être appelée plusieurs fois pour qu'une même entité effectue plusieurs lignes de configuration.

3

HasKey<TKey>

Configure la ou les propriétés de clé primaire pour ce type d'entité.

4

HasMany<TTargetEntity>

Configure plusieurs relations à partir de ce type d'entité.

5

HasOptional<TTargetEntity>

Configure une relation facultative à partir de ce type d'entité. Les instances du type d'entité pourront être enregistrées dans la base de données sans que cette relation ne soit spécifiée. La clé étrangère dans la base de données sera Nullable.

6

HasRequired<TTargetEntity>

Configure une relation requise à partir de ce type d'entité. Les instances du type d'entité ne pourront pas être enregistrées dans la base de données à moins que cette relation ne soit spécifiée. La clé étrangère dans la base de données sera non nullable.

sept

Ignore<TProperty>

Exclut une propriété du modèle afin qu'elle ne soit pas mappée à la base de données. (Hérité de StructuralTypeConfiguration <TStructuralType>)

8

Property<T>

Configure une propriété struct définie sur ce type. (Hérité de StructuralTypeConfiguration <TStructuralType>)

9

ToTable(String)

Configure le nom de table auquel ce type d'entité est mappé.

L'API Fluent vous permet de configurer vos entités ou leurs propriétés, que vous souhaitiez modifier la manière dont elles sont mappées à la base de données ou comment elles sont liées les unes aux autres. Il existe une grande variété de mappages et de modèles sur lesquels vous pouvez avoir un impact en utilisant les configurations. Voici les principaux types de mappage pris en charge par Fluent API -

  • Mappage d'entités
  • Cartographie des propriétés

Mappage d'entités

Le mappage d'entité n'est que quelques mappages simples qui auront un impact sur la compréhension d'Entity Framework de la façon dont les classes sont mappées aux bases de données. Nous avons discuté de tout cela dans les annotations de données et nous verrons ici comment réaliser les mêmes choses en utilisant Fluent API.

  • Donc, plutôt que d'aller dans les classes de domaine pour ajouter ces configurations, nous pouvons le faire à l'intérieur du contexte.

  • La première chose est de remplacer la méthode OnModelCreating, qui permet au modelBuilder de travailler avec.

Schéma par défaut

Le schéma par défaut est dbo lorsque la base de données est générée. Vous pouvez utiliser la méthode HasDefaultSchema sur DbModelBuilder pour spécifier le schéma de base de données à utiliser pour toutes les tables, procédures stockées, etc.

Jetons un coup d'œil à l'exemple suivant dans lequel le schéma d'administration est appliqué.

public class MyContext : DbContext {
   public MyContext() : base("name = MyContextDB") {}

   protected override void OnModelCreating(DbModelBuilder modelBuilder) {
      //Configure default schema
      modelBuilder.HasDefaultSchema("Admin");
   }
	
   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

Mapper l'entité à la table

Avec la convention par défaut, Code First créera les tables de base de données avec le nom des propriétés DbSet dans la classe de contexte, telles que Cours, Inscriptions et Etudiants. Mais si vous souhaitez des noms de table différents, vous pouvez remplacer cette convention et fournir un nom de table différent de celui des propriétés DbSet, comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   //Map entity to table
   modelBuilder.Entity<Student>().ToTable("StudentData");
   modelBuilder.Entity<Course>().ToTable("CourseDetail");
   modelBuilder.Entity<Enrollment>().ToTable("EnrollmentInfo");
}

Lorsque la base de données est générée, vous verrez le nom des tables tel que spécifié dans la méthode OnModelCreating.

Fractionnement d'entité (mapper l'entité à plusieurs tables)

Le fractionnement d'entités vous permet de combiner des données provenant de plusieurs tables en une seule classe et il ne peut être utilisé qu'avec des tables qui ont une relation un-à-un entre elles. Jetons un coup d'œil à l'exemple suivant dans lequel les informations sur les étudiants sont mappées dans deux tables.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {
   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   //Map entity to table
   modelBuilder.Entity<Student>().Map(sd ⇒ {
      sd.Properties(p ⇒ new { p.ID, p.FirstMidName, p.LastName });
      sd.ToTable("StudentData");
   })

   .Map(si ⇒ {
      si.Properties(p ⇒ new { p.ID, p.EnrollmentDate });
      si.ToTable("StudentEnrollmentInfo");
   });

   modelBuilder.Entity<Course>().ToTable("CourseDetail");
   modelBuilder.Entity<Enrollment>().ToTable("EnrollmentInfo");
}

Dans le code ci-dessus, vous pouvez voir que l'entité Student est divisée en deux tables suivantes en mappant certaines propriétés à la table StudentData et certaines propriétés à la table StudentEnrollmentInfo à l'aide de la méthode Map.

  • StudentData - Contient le prénom et le nom de l'étudiant.

  • StudentEnrollmentInfo - Contient la date d'inscription.

Lorsque la base de données est générée, vous voyez les tables suivantes dans votre base de données, comme illustré dans l'image suivante.

Cartographie des propriétés

La méthode Property est utilisée pour configurer les attributs de chaque propriété appartenant à une entité ou à un type complexe. La méthode Property est utilisée pour obtenir un objet de configuration pour une propriété donnée. Vous pouvez également mapper et configurer les propriétés de vos classes de domaine à l'aide de l'API Fluent.

Configurer une clé primaire

La convention par défaut pour les clés primaires est -

  • La classe définit une propriété dont le nom est «ID» ou «Id»
  • Nom du cours suivi de "ID" ou "Id"

Si votre classe ne suit pas les conventions par défaut pour la clé primaire comme indiqué dans le code suivant de la classe Student -

public class Student {
   public int StdntID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Ensuite, pour définir explicitement une propriété comme clé primaire, vous pouvez utiliser la méthode HasKey comme indiqué dans le code suivant -

protected override void OnModelCreating(DbModelBuilder modelBuilder) {
   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");
	
   // Configure Primary Key
   modelBuilder.Entity<Student>().HasKey<int>(s ⇒ s.StdntID); 
}

Configurer la colonne

Dans Entity Framework, par défaut Code First crée une colonne pour une propriété avec le même nom, ordre et type de données. Mais vous pouvez également remplacer cette convention, comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   //Configure EnrollmentDate Column
   modelBuilder.Entity<Student>().Property(p ⇒ p.EnrollmentDate)
	
   .HasColumnName("EnDate")
   .HasColumnType("DateTime")
   .HasColumnOrder(2);
}

Configurer la propriété MaxLength

Dans l'exemple suivant, la propriété Titre du cours ne doit pas dépasser 24 caractères. Lorsque l'utilisateur spécifie une valeur de plus de 24 caractères, l'utilisateur obtient une exception DbEntityValidationException.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {
   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");
   modelBuilder.Entity<Course>().Property(p ⇒ p.Title).HasMaxLength(24);
}

Configurer la propriété Null ou NotNull

Dans l'exemple suivant, la propriété Course Title est obligatoire afin que la méthode IsRequired soit utilisée pour créer la colonne NotNull. De même, Student EnrollmentDate est facultatif, nous utiliserons donc la méthode IsOptional pour autoriser une valeur nulle dans cette colonne, comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");
   modelBuilder.Entity<Course>().Property(p ⇒ p.Title).IsRequired();
   modelBuilder.Entity<Student>().Property(p ⇒ p.EnrollmentDate).IsOptional();
	
   //modelBuilder.Entity<Student>().Property(s ⇒ s.FirstMidName)
   //.HasColumnName("FirstName"); 
}

Configurer les relations

Une relation, dans le contexte des bases de données, est une situation qui existe entre deux tables de base de données relationnelle, lorsqu'une table a une clé étrangère qui fait référence à la clé primaire de l'autre table. Lorsque vous travaillez avec Code First, vous définissez votre modèle en définissant vos classes CLR de domaine. Par défaut, Entity Framework utilise les conventions Code First pour mapper vos classes au schéma de base de données.

  • Si vous utilisez les conventions de dénomination Code First, dans la plupart des cas, vous pouvez compter sur Code First pour configurer les relations entre vos tables en fonction des clés étrangères et des propriétés de navigation.

  • S'ils ne respectent pas ces conventions, vous pouvez également utiliser des configurations pour avoir un impact sur les relations entre les classes et sur la manière dont ces relations sont réalisées dans la base de données lorsque vous ajoutez des configurations dans Code First.

  • Certains d'entre eux sont disponibles dans les annotations de données et vous pouvez en appliquer d'autres encore plus complexes avec une API Fluent.

Configurer la relation un-à-un

Lorsque vous définissez une relation un-à-un dans votre modèle, vous utilisez une propriété de navigation de référence dans chaque classe. Dans la base de données, les deux tables ne peuvent avoir qu'un seul enregistrement de chaque côté de la relation. Chaque valeur de clé primaire se rapporte à un seul enregistrement (ou aucun enregistrement) dans la table associée.

  • Une relation un-à-un est créée si les deux colonnes associées sont des clés primaires ou ont des contraintes uniques.

  • Dans une relation un-à-un, la clé primaire agit en outre comme une clé étrangère et il n'y a pas de colonne de clé étrangère distincte pour l'une ou l'autre table.

  • Ce type de relation n'est pas courant car la plupart des informations ainsi liées se trouveraient toutes dans un seul tableau.

Jetons un coup d'œil à l'exemple suivant où nous ajouterons une autre classe dans notre modèle pour créer une relation un-à-un.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual StudentLogIn StudentLogIn { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

public class StudentLogIn {
   [Key, ForeignKey("Student")]
   public int ID { get; set; }
   public string EmailID { get; set; }
   public string Password { get; set; }
	
   public virtual Student Student { get; set; }
}

Comme vous pouvez le voir dans le code ci-dessus, les attributs Key et ForeignKey sont utilisés pour la propriété ID dans la classe StudentLogIn, afin de le marquer comme clé primaire ainsi que comme clé étrangère.

Pour configurer une relation un à zéro ou une relation entre Student et StudentLogIn à l'aide de l'API Fluent, vous devez remplacer la méthode OnModelCreating comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   // Configure ID as PK for StudentLogIn
   modelBuilder.Entity<StudentLogIn>()
   .HasKey(s ⇒ s.ID);

   // Configure ID as FK for StudentLogIn
   modelBuilder.Entity<Student>()
   
   .HasOptional(s ⇒ s.StudentLogIn) //StudentLogIn is optional
   .WithRequired(t ⇒ t.Student); // Create inverse relationship
}

Dans la plupart des cas, Entity Framework peut déduire quel type est le dépendant et quel est le principal dans une relation. Cependant, lorsque les deux extrémités de la relation sont requises ou que les deux côtés sont facultatifs, Entity Framework ne peut pas identifier la personne à charge et le mandant. Lorsque les deux extrémités de la relation sont requises, vous pouvez utiliser HasRequired comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   // Configure ID as PK for StudentLogIn
   modelBuilder.Entity<StudentLogIn>()
   .HasKey(s ⇒ s.ID);

   // Configure ID as FK for StudentLogIn
   modelBuilder.Entity<Student>()
   .HasRequired(r ⇒ r.Student)
   .WithOptional(s ⇒ s.StudentLogIn);  
}

Lorsque la base de données est générée, vous verrez que la relation est créée comme indiqué dans l'image suivante.

Configurer la relation un-à-plusieurs

La table de clé primaire contient un seul enregistrement qui ne concerne aucun, un ou plusieurs enregistrements de la table associée. C'est le type de relation le plus couramment utilisé.

  • Dans ce type de relation, une ligne de la table A peut avoir plusieurs lignes correspondantes dans la table B, mais une ligne de la table B ne peut avoir qu'une seule ligne correspondante dans la table A.

  • La clé étrangère est définie sur la table qui représente l'extrémité plusieurs de la relation.

  • Par exemple, dans le diagramme ci-dessus, les tables Étudiant et Inscription ont une relation un à plusieurs, chaque étudiant peut avoir plusieurs inscriptions, mais chaque inscription appartient à un seul étudiant.

Vous trouverez ci-dessous l'étudiant et l'inscription qui ont une relation un-à-plusieurs, mais la clé étrangère dans la table d'inscription ne suit pas les conventions Code First par défaut.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
	
   //StdntID is not following code first conventions name
   public int StdntID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }
}

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual StudentLogIn StudentLogIn { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Dans ce cas, pour configurer une relation un-à-plusieurs à l'aide de l'API Fluent, vous devez utiliser la méthode HasForeignKey comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   //Configure FK for one-to-many relationship
   modelBuilder.Entity<Enrollment>()

   .HasRequired<Student>(s ⇒ s.Student)
   .WithMany(t ⇒ t.Enrollments)
   .HasForeignKey(u ⇒ u.StdntID);  
}

Lorsque la base de données est générée, vous verrez que la relation est créée comme indiqué dans l'image suivante.

Dans l'exemple ci-dessus, la méthode HasRequired spécifie que la propriété de navigation Student doit être Null. Vous devez donc attribuer une entité Étudiant avec inscription à chaque fois que vous ajoutez ou mettez à jour l'inscription. Pour gérer cela, nous devons utiliser la méthode HasOptional au lieu de la méthode HasRequired.

Configurer la relation plusieurs-à-plusieurs

Chaque enregistrement dans les deux tables peut se rapporter à n'importe quel nombre d'enregistrements (ou aucun enregistrement) dans l'autre table.

  • Vous pouvez créer une telle relation en définissant une troisième table, appelée table de jonction, dont la clé primaire est constituée des clés étrangères de la table A et de la table B.

  • Par exemple, la table Student et la table Course ont une relation plusieurs-à-plusieurs.

Vous trouverez ci-dessous les classes Etudiant et Cours dans lesquelles Etudiant et Cours ont une relation plusieurs à plusieurs, car les deux classes ont des propriétés de navigation Etudiants et Cours qui sont des collections. En d'autres termes, une entité a une autre collection d'entités.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Course> Courses { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

public class Course {
   public int CourseID { get; set; }
   public string Title { get; set; }
   public int Credits { get; set; }
	
   public virtual ICollection<Student> Students { get; set; }
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

Pour configurer la relation plusieurs-à-plusieurs entre l'étudiant et le cours, vous pouvez utiliser l'API Fluent comme indiqué dans le code suivant.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   // Configure many-to-many relationship
   modelBuilder.Entity<Student>()
   .HasMany(s ⇒ s.Courses) 
   .WithMany(s ⇒ s.Students);
}

Les conventions Code First par défaut sont utilisées pour créer une table de jointure lorsque la base de données est générée. Par conséquent, la table StudentCourses est créée avec les colonnes Course_CourseID et Student_ID, comme illustré dans l'image suivante.

Si vous souhaitez spécifier le nom de la table de jointure et les noms des colonnes de la table, vous devez effectuer une configuration supplémentaire à l'aide de la méthode Map.

protected override void OnModelCreating(DbModelBuilder modelBuilder) {

   //Configure default schema
   modelBuilder.HasDefaultSchema("Admin");

   // Configure many-to-many relationship 
   modelBuilder.Entity<Student>()

   .HasMany(s ⇒ s.Courses)
   .WithMany(s ⇒ s.Students)
   
   .Map(m ⇒ {
      m.ToTable("StudentCoursesTable");
      m.MapLeftKey("StudentID");
      m.MapRightKey("CourseID");
   }); 
}

Vous pouvez voir quand la base de données est générée, le nom de la table et des colonnes est créé comme spécifié dans le code ci-dessus.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans Entity Framework, Seed a été introduit dans EF 4.1 et fonctionne avec les initialiseurs de base de données. L'idée générale d'unSeed Methodconsiste à initialiser les données dans une base de données créée par Code First ou évoluée par Migrations. Ces données sont souvent des données de test, mais peuvent également être des données de référence telles que des listes d'étudiants connus, des cours, etc. Lorsque les données sont initialisées, il effectue les opérations suivantes -

  • Vérifie si la base de données cible existe déjà ou non.
  • Si tel est le cas, le modèle Code First actuel est comparé au modèle stocké dans les métadonnées de la base de données.
  • La base de données est supprimée si le modèle actuel ne correspond pas au modèle de la base de données.
  • La base de données est créée si elle a été supprimée ou n'existait pas en premier lieu.
  • Si la base de données a été créée, la méthode de l'initialiseur Seed est appelée.

La méthode Seed prend l'objet de contexte de base de données comme paramètre d'entrée et le code de la méthode utilise cet objet pour ajouter de nouvelles entités à la base de données. Pour amorcer des données dans votre base de données, vous devez remplacer la méthode Seed. Jetons un œil à l'exemple suivant dans lequel certaines des données par défaut sont initiées dans la base de données dans une classe interne.

private class UniDBInitializer<T> : DropCreateDatabaseAlways<MyContext> {

   protected override void Seed(MyContext context) {

      IList<Student> students = new List<Student>();

      students.Add(new Student() {
         FirstMidName = "Andrew", 
         LastName = "Peters", 
         EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
      });

      students.Add(new Student() {
         FirstMidName = "Brice", 
         LastName = "Lambson", 
         EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
      });

      students.Add(new Student() {
         FirstMidName = "Rowan", 
         LastName = "Miller", 
         EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
      });

      foreach (Student student in students)
      context.Students.Add(student);
      base.Seed(context);
   }
}

Dans le code ci-dessus, la table des étudiants est initialisée. Vous devez définir cette classe d'initialisation de base de données dans la classe de contexte, comme indiqué dans le code suivant.

public MyContext() : base("name=MyContextDB") {
   Database.SetInitializer<MyContext>(new UniDBInitializer<MyContext>());
}

Voici l'implémentation de classe complète de la classe MyContext, qui contient également la classe d'initialisation DB.

public class MyContext : DbContext {

   public MyContext() : base("name=MyContextDB") {
      Database.SetInitializer<MyContext>(new UniDBInitializer<MyContext>());
   }

   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
	
   private class UniDBInitializer<T> : DropCreateDatabaseAlways<MyContext> {

      protected override void Seed(MyContext context) {

         IList<Student> students = new List<Student>();
			
         students.Add(new Student() {
            FirstMidName = "Andrew", 
            LastName = "Peters", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString()) 
         });

         students.Add(new Student() {
            FirstMidName = "Brice", 
            LastName = "Lambson", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         });

         students.Add(new Student() {
            FirstMidName = "Rowan", 
            LastName = "Miller", 
            EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         });

         foreach (Student student in students)
         context.Students.Add(student);
         base.Seed(context);
      }
   } 
}

Lorsque l'exemple ci-dessus est compilé et exécuté, vous pouvez voir les données dans une base de données comme indiqué dans l'image suivante.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Entity Framework 4.3 inclut une nouvelle fonctionnalité de migrations Code First qui vous permet de faire évoluer progressivement le schéma de base de données à mesure que votre modèle change au fil du temps. Pour la plupart des développeurs, il s'agit d'une grande amélioration par rapport aux options d'initialisation de base de données des versions 4.1 et 4.2 qui vous obligeaient à mettre à jour manuellement la base de données ou à la supprimer et à la recréer lorsque votre modèle a changé.

  • Avant Entity Framework 4.3, si vous avez déjà des données (autres que des données de départ) ou des procédures stockées, des déclencheurs, etc. existants dans votre base de données, ces stratégies permettaient de supprimer toute la base de données et de la recréer, de sorte que vous perdriez les données et les autres bases de données. objets.

  • Avec la migration, il mettra automatiquement à jour le schéma de la base de données, lorsque votre modèle change sans perdre de données existantes ou d'autres objets de base de données.

  • Il utilise un nouvel initialiseur de base de données appelé MigrateDatabaseToLatestVersion.

Il existe deux types de migration -

  • Migration automatisée
  • Migration basée sur le code

Migration automatisée

La migration automatisée a été introduite pour la première fois dans Entity Framework 4.3. Dans la migration automatisée, vous n'avez pas besoin de traiter la migration de la base de données manuellement dans le fichier de code. Par exemple, pour chaque modification, vous devrez également modifier vos classes de domaine. Mais avec la migration automatisée, il vous suffit d'exécuter une commande dans la console du gestionnaire de package pour y parvenir.

Jetons un coup d'œil au processus étape par étape suivant de migration automatisée.

Lorsque vous utilisez l'approche Code First, vous n'avez pas de base de données pour votre application.

Dans cet exemple, nous commencerons par nos 3 classes de base telles que les étudiants, les cours et les inscriptions, comme indiqué dans le code suivant.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }

}

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

public class Course {
   public int CourseID { get; set; }
   public string Title { get; set; }
   [Index]
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

Voici la classe de contexte.

public class MyContext : DbContext {
   public MyContext() : base("MyContextDB") {}
   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }
}

Avant d'exécuter l'application, vous devez activer la migration automatisée.

Step 1 - Ouvrez la console du gestionnaire de packages depuis Outils → Gestionnaire de packages NuGet → Console du gestionnaire de packages.

Step 2 - Pour activer la migration automatisée, exécutez la commande suivante dans la console du gestionnaire de package.

PM> enable-migrations -EnableAutomaticMigrations:$true

Step 3 - Une fois que la commande s'exécute avec succès, elle crée une classe de configuration interne scellée dans le dossier Migration de votre projet, comme indiqué dans le code suivant.

namespace EFCodeFirstDemo.Migrations {

   using System;
   using System.Data.Entity;
   using System.Data.Entity.Migrations;
   using System.Linq;
	
   internal sealed class Configuration : DbMigrationsConfiguration<EFCodeFirstDemo.MyContext> {

      public Configuration() {
         AutomaticMigrationsEnabled = true;
         ContextKey = "EFCodeFirstDemo.MyContext";
      }

      protected override void Seed(EFCodeFirstDemo.MyContext context) {

         //  This method will be called after migrating to the latest version.
         //  You can use the DbSet<T>.AddOrUpdate() helper extension method
         //  to avoid creating duplicate seed data. E.g.

         //  context.People.AddOrUpdate(
            //  p ⇒ p.FullName, 
            //  new Person { FullName = "Andrew Peters" }, 
            //  new Person { FullName = "Brice Lambson" }, 
            //  new Person { FullName = "Rowan Miller" }
         //  );
      }
   }
}

Step 4 - Définissez l'initialiseur de base de données dans la classe de contexte avec la nouvelle stratégie d'initialisation de base de données MigrateDatabaseToLatestVersion.

public class MyContext : DbContext {

   public MyContext() : base("MyContextDB") {
      Database.SetInitializer(new MigrateDatabaseToLatestVersion<MyContext, 
         EFCodeFirstDemo.Migrations.Configuration>("MyContextDB"));
   }

   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }

}

Step 5- Vous avez mis en place une migration automatisée. Lorsque vous exécutez votre application, celle-ci se charge automatiquement de la migration, lorsque vous modifiez le modèle.

Step 6- Comme vous pouvez le voir, une table système __MigrationHistory est également créée dans votre base de données avec d'autres tables. Dans __MigrationHistory, la migration automatisée conserve l'historique des modifications de la base de données.

Step 7- Lorsque vous ajoutez une autre classe d'entité en tant que classe de votre domaine et exécutez votre application, elle crée la table dans votre base de données. Ajoutons la classe StudentLogIn suivante.

public class StudentLogIn {
   [Key, ForeignKey("Student")]
   public int ID { get; set; }
   public string EmailID { get; set; }
   public string Password { get; set; }
	
   public virtual Student Student { get; set; }
}

Step 8 - N'oubliez pas d'ajouter le DBSet pour la classe mentionnée ci-dessus dans votre classe de contexte comme indiqué dans le code suivant.

public virtual DbSet<StudentLogIn> StudentsLogIn { get; set; }

Step 9 - Exécutez à nouveau votre application et vous verrez que la table StudentsLogIn est ajoutée à votre base de données.

Les étapes ci-dessus mentionnées pour les migrations automatisées ne fonctionneront que pour votre entité. Par exemple, pour ajouter une autre classe d'entité ou supprimer la classe d'entité existante, la migration réussira. Mais si vous ajoutez ou supprimez une propriété à votre classe d'entité, elle lèvera une exception.

Step 10 - Pour gérer la migration des propriétés, vous devez définir AutomaticMigrationDataLossAllowed = true dans le constructeur de la classe de configuration.

public Configuration() {
   AutomaticMigrationsEnabled = true;
   AutomaticMigrationDataLossAllowed = true;
   ContextKey = "EFCodeFirstDemo.MyContext";
}

Migration basée sur le code

Lorsque vous développez une nouvelle application, votre modèle de données change fréquemment et à chaque fois que le modèle change, il se désynchronise avec la base de données. Vous avez configuré Entity Framework pour supprimer et recréer automatiquement la base de données chaque fois que vous modifiez le modèle de données. La migration basée sur le code est utile lorsque vous souhaitez plus de contrôle sur la migration.

  • Lorsque vous ajoutez, supprimez ou modifiez des classes d'entité ou que vous modifiez votre classe DbContext, la prochaine fois que vous exécutez l'application, elle supprime automatiquement votre base de données existante, en crée une nouvelle qui correspond au modèle et l'amorce avec des données de test.

  • La fonctionnalité Migrations Code First résout ce problème en permettant à Code First de mettre à jour le schéma de base de données au lieu de supprimer et de recréer la base de données. Pour déployer l'application, vous devrez activer les migrations.

Voici la règle de base pour migrer les modifications dans la base de données -

  • Activer les migrations
  • Ajouter une migration
  • Mettre à jour la base de données

Jetons un coup d'œil au processus étape par étape suivant de migration de base de code.

Lorsque vous utilisez d'abord le code, vous n'avez pas de base de données pour votre application.

Dans cet exemple, nous recommencerons avec nos 3 classes de base telles que Etudiant, Cours et Inscription, comme indiqué dans le code suivant.

public class Enrollment {
   public int EnrollmentID { get; set; }
   public int CourseID { get; set; }
   public int StudentID { get; set; }
   public Grade? Grade { get; set; }
	
   public virtual Course Course { get; set; }
   public virtual Student Student { get; set; }

}

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

public class Course {
   public int CourseID { get; set; }
   public string Title { get; set; }
   [Index]
   public int Credits { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

Voici la classe de contexte.

public class MyContext : DbContext {

   public MyContext() : base("MyContextDB") {
      Database.SetInitializer(new MigrateDatabaseToLatestVersion<
         MyContext, EFCodeFirstDemo.Migrations.Configuration>("MyContextDB"));
   }

   public virtual DbSet<Course> Courses { get; set; }
   public virtual DbSet<Enrollment> Enrollments { get; set; }
   public virtual DbSet<Student> Students { get; set; }

}

Step 1 - Avant d'exécuter l'application, vous devez activer la migration.

Step 2 - Ouvrez la console du gestionnaire de packages depuis Outils → Gestionnaire de packages NuGet → Console du gestionnaire de packages.

Step 3 - La migration est déjà activée, ajoutez maintenant la migration dans votre application en exécutant la commande suivante.

PM> add-migration "UniDB Schema"

Step 4 - Lorsque la commande est exécutée avec succès, vous verrez qu'un nouveau fichier a été créé dans le dossier Migration avec le nom du paramètre que vous avez passé à la commande avec un préfixe d'horodatage comme indiqué dans l'image suivante.

Step 5 - Vous pouvez créer ou mettre à jour la base de données à l'aide de la commande «update-database».

PM> Update-Database -Verbose

L'indicateur "-Verbose" spécifie d'afficher les instructions SQL appliquées à la base de données cible dans la console.

Step 6 - Ajoutons une autre propriété 'Age' dans la classe étudiante, puis exécutons l'instruction de mise à jour.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public int Age { get; set; }
   public DateTime EnrollmentDate { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

}

Lorsque vous exécutez PM → Update-Database –Verbose, lorsque la commande est exécutée avec succès, vous verrez que la nouvelle colonne Age est ajoutée dans votre base de données.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Dans ce chapitre, nous allons apprendre à migrer les modifications dans la base de données lorsqu'il y a plusieurs classes DbContext dans l'application.

  • Multiple DbContext a été introduit pour la première fois dans Entity Framework 6.0.
  • Plusieurs classes de contexte peuvent appartenir à une seule base de données ou à deux bases de données différentes.

Dans notre exemple, nous définirons deux classes de contexte pour la même base de données. Dans le code suivant, il existe deux classes DbContext pour Student et Teacher.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
}

public class MyStudentContext : DbContext {
   public MyStudentContext() : base("UniContextDB") {}
   public virtual DbSet<Student> Students { get; set; }
}

public class Teacher {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime HireDate { get; set; }
}

public class MyTeacherContext : DbContext {
   public MyTeacherContext() : base("UniContextDB") {}
   public virtual DbSet<Teacher> Teachers { get; set; }
}

Comme vous pouvez le voir dans le code ci-dessus, il existe deux modèles appelés «étudiant» et «enseignant». Chacun est associé à une classe de contexte correspondante particulière, c'est-à-dire que Student est associé à MyStudentContext et Teacher est associé à MyTeacherContext.

Voici la règle de base pour migrer les modifications dans la base de données, lorsqu'il existe plusieurs classes de contexte dans le même projet.

  • enable-migrations -ContextTypeName <DbContext-Name-with-Namespaces> MigrationsDirectory: <Migrations-Directory-Name>

  • Add-Migration -configuration <DbContext-Migrations-Configuration-Class-withNamespaces> <Migrations-Name>

  • Update-Database -configuration <DbContext-Migrations-Configuration-Class-withNamespaces> -Verbose

Activons la migration pour MyStudentContext en exécutant la commande suivante dans la console du gestionnaire de package.

PM→ enable-migrations -ContextTypeName:EFCodeFirstDemo.MyStudentContext

Une fois qu'il sera exécuté, nous ajouterons le modèle dans l'historique de migration et pour cela, nous devons lancer la commande add-migration dans la même console.

PM→ add-migration -configuration EFCodeFirstDemo.Migrations.Configuration Initial

Ajoutons maintenant des données dans les tableaux Elèves et Professeurs de la base de données.

static void Main(string[] args) {

   using (var context = new MyStudentContext()) {
	
      //// Create and save a new Students
      Console.WriteLine("Adding new students");

      var student = new Student {
         FirstMidName = "Alain", 
         LastName = "Bomer", 
         EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         //Age = 24
      };

      context.Students.Add(student);

      var student1 = new Student {
         FirstMidName = "Mark",
         LastName = "Upston", 
         EnrollmentDate = DateTime.Parse(DateTime.Today.ToString())
         //Age = 30
      };

      context.Students.Add(student1);
      context.SaveChanges();
      // Display all Students from the database
      var students = (from s in context.Students orderby s.FirstMidName
         select s).ToList<Student>();
		
      Console.WriteLine("Retrieve all Students from the database:");

      foreach (var stdnt in students) {
         string name = stdnt.FirstMidName + " " + stdnt.LastName;
         Console.WriteLine("ID: {0}, Name: {1}", stdnt.ID, name);
      }

      Console.WriteLine("Press any key to exit...");
      Console.ReadKey();
   }

   using (var context = new MyTeacherContext()) {

      //// Create and save a new Teachers
      Console.WriteLine("Adding new teachers");

      var student = new Teacher {
         FirstMidName = "Alain", 
         LastName = "Bomer", 
         HireDate = DateTime.Parse(DateTime.Today.ToString())
         //Age = 24
      };

      context.Teachers.Add(student);

      var student1 = new Teacher {
         FirstMidName = "Mark", 
         LastName = "Upston", 
         HireDate = DateTime.Parse(DateTime.Today.ToString())
         //Age = 30
      };

      context.Teachers.Add(student1);
      context.SaveChanges();
  
      // Display all Teachers from the database
      var teachers = (from t in context.Teachers orderby t.FirstMidName
         select t).ToList<Teacher>();
		
      Console.WriteLine("Retrieve all teachers from the database:");

      foreach (var teacher in teachers) {
         string name = teacher.FirstMidName + " " + teacher.LastName;
         Console.WriteLine("ID: {0}, Name: {1}", teacher.ID, name);
      }

      Console.WriteLine("Press any key to exit...");
      Console.ReadKey();
   }
}

Lorsque le code ci-dessus est exécuté, vous verrez que deux tables différentes sont créées pour deux modèles différents, comme illustré dans l'image suivante.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.

Avant Entity Framework 6, Entity Framework ne reconnaissait pas les entités ou les types complexes imbriqués dans d'autres entités ou types complexes. Lorsque Entity Framework a généré le modèle, les types imbriqués ont simplement disparu.

Jetons un coup d'œil à un exemple simple dans lequel nous avons notre modèle de base avec trois entités Étudiant, Cours et Inscription.

  • Ajoutons une propriété Identity, qui est un type Person. Person est une autre entité, contient les propriétés BirthDate et FatherName.

  • En termes d'Entity Framework, comme il n'a pas d'identité et fait partie d'une entité, il s'agit d'un type complexe Entity Framework, et nous avons en fait pris en charge les types complexes depuis la première version d'Entity Framework.

  • Le type Person n'est pas imbriqué comme indiqué dans le code suivant.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
   public DateTime EnrollmentDate { get; set; }
   public Person Identity { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }
}

public class Person {

   public Person(string fatherName, DateTime birthDate) {
      FatherName = fatherName;
      BirthDate = birthDate;
   }
	
   public string FatherName { get; set; }
   public DateTime BirthDate { get; set; }
}

Entity Framework saura également comment conserver les types de personne lorsqu'il est utilisé dans les versions précédentes.

En utilisant Entity Framework Power Tool, nous verrons comment Entity Framework interprète le modèle. Cliquez avec le bouton droit sur le fichier Program.cs et sélectionnez Entity Framework → Afficher le modèle de données d'entité (lecture seule)

Vous verrez maintenant que la propriété Identity est définie dans la classe Student.

Si cette classe Person ne sera utilisée par aucune autre entité, nous pouvons l'imbriquer dans la classe Student, mais cette version antérieure d'Entity Framework ne reconnaît pas les types imbriqués.

Dans les anciennes versions, vous générez à nouveau le modèle, non seulement le type n'est pas reconnu, mais parce qu'il n'y est pas, la propriété n'y est pas non plus, donc Entity Framework ne persistera pas du tout le type Person.

public class Student {
   public int ID { get; set; }
   public string LastName { get; set; }
   public string FirstMidName { get; set; }
	
   public DateTime EnrollmentDate { get; set; }
   public Person Identity { get; set; }
	
   public virtual ICollection<Enrollment> Enrollments { get; set; }

   public class Person {

      public Person(string fatherName, DateTime birthDate) {
         FatherName = fatherName;
         BirthDate = birthDate;
      }

      public string FatherName { get; set; }
      public DateTime BirthDate { get; set; }
   }
}

Avec Entity Framework 6, les entités imbriquées et les types complexes sont reconnus. Dans le code ci-dessus, vous pouvez voir que Person est imbriqué dans la classe Student.

Lorsque vous utilisez l'outil d'alimentation Entity Framework pour montrer comment Entity Framework interprète le modèle cette fois, il existe une véritable propriété Identity et un type complexe Person. Donc Entity Framework conservera ces données.

Vous pouvez maintenant voir que Identity est un type d'entité imbriqué, qui n'était pas pris en charge avant Entity Framework 6.

Nous vous recommandons d'exécuter l'exemple ci-dessus étape par étape pour une meilleure compréhension.