NHibernate - Cascades
Dans ce chapitre, nous expliquerons comment utiliser la fonction Cascade. Si vous avez un ensemble ou une collection d'articles ou une relation entre deux classes telles que notre client et notre commande et avez une relation de clé étrangère. Si nous supprimons le client par défaut, NHibernate ne fait rien aux objets enfants, donc ceux qui appartiennent à ce client et nous pourrions être des commandes orphelines.
Nous pourrions également violer les contraintes de clé étrangère, nous pouvons donc utiliser la notion de cascades.
Par défaut, NHibernate ne cascade pas les opérations sur les objets enfants.
La raison en est que vous pouvez avoir des relations telles qu'un client ayant une adresse de livraison par défaut et cette adresse de livraison est partagée avec de nombreux clients différents.
Vous ne voudriez donc pas nécessairement mettre en cascade cette relation parce que d'autres clients y font encore référence.
Toute la notion de cascades est donc de dire à NHibernate comment gérer ses entités enfants.
Il existe différentes options pour la mise en cascade, qui sont les suivantes -
none - qui est la valeur par défaut et cela signifie pas de cascade.
all - qui va enregistrer en cascade, mettre à jour et supprimer.
save-update - il va en cascade, enregistre et met à jour.
delete - il supprimera en cascade.
all-delete-orphan - c'est un système spécial qui est assez fréquemment utilisé et qui est le même que Tous sauf, s'il trouve des lignes Delete-orphan, il les supprimera également.
Vous pouvez spécifier la valeur par défaut dans votre hbm.xml afin que vous puissiez fournir une cascade par défaut sur cet élément de mappage Hibernate ou vous pouvez également le spécifier pour des collections et des relations spécifiques telles que plusieurs-à-un.
Jetons un coup d'œil à de simples exemples de cascades, corrigeons le problème dans le programme, où nous devons manuellement cascader la sauvegarde aux commandes comme indiqué dans le code suivant.
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
var newCustomer = CreateCustomer();
Console.WriteLine("New Customer:");
Console.WriteLine(newCustomer);
session.Save(newCustomer);
foreach (var order in newCustomer.Orders) {
session.Save(order);
}
id = newCustomer.Id;
tx.Commit();
}
Dans l'extrait de code ci-dessus, vous pouvez voir que nous enregistrons manuellement toutes les commandes du client. Supprimons maintenant le code de cascade manuel dans lequel toutes les commandes sont enregistrées.
using(var session = sessionFactory.OpenSession())
using(var tx = session.BeginTransaction()) {
var newCustomer = CreateCustomer();
Console.WriteLine("New Customer:");
Console.WriteLine(newCustomer);
session.Save(newCustomer);
id = newCustomer.Id;
tx.Commit();
}
Nous devons spécifier l'option cascade dans customer.hbm.xml.
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly = "NHibernateDemo"
namespace = "NHibernateDemo">
<class name = "Customer">
<id name = "Id">
<generator class = "guid.comb"/>
</id>
<property name = "FirstName"/>
<property name = "LastName"/>
<property name = "AverageRating"/>
<property name = "Points"/>
<property name = "HasGoldStatus"/>
<property name = "MemberSince" type = "UtcDateTime"/>
<property name = "CreditRating" type = "CustomerCreditRatingType"/>
<component name = "Address">
<property name = "Street"/>
<property name = "City"/>
<property name = "Province"/>
<property name = "Country"/>
</component>
<set name = "Orders" table = "`Order`" cascade = "all-delete-orphan">
<key column = "CustomerId"/>
<one-to-many class = "Order"/>
</set>
</class>
</hibernate-mapping>
Désormais, les commandes appartiennent entièrement au client. Donc, si les clients étaient supprimés de la base de données, notre application voudrait supprimer toutes ces commandes, y compris celles qui auraient pu être orphelines.
Il finira par faire une suppression. Par cela, il indiquera supprimer de la table de commande, où l'ID client correspond au client que vous supprimez.
Vous pouvez donc effectivement mettre en cascade ces suppressions. Donc avec leAll, il effectuera des sauvegardes, des mises à jour et des suppressions.
Maintenant, lorsque vous exécutez cette application, vous verrez la sortie suivante.
New Customer:
John Doe (00000000-0000-0000-0000-000000000000)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Unspecified)
CreditRating: Good
AverageRating: 42.42424242
Orders:
Order Id: 00000000-0000-0000-0000-000000000000
Order Id: 00000000-0000-0000-0000-000000000000
Reloaded:
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
The orders were ordered by:
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
Press <ENTER> to exit...
Comme vous pouvez le voir, nous avons supprimé le code du programme qui est manuellement en cascade et notre application fonctionne toujours.
Donc, selon votre relation, vous voudrez peut-être les mettre en cascade. Voyons maintenant une relation en cascade différente. Allons auOrder.hbm.xml fichier et nous pouvons mettre en cascade cette relation plusieurs-à-un.
<?xml version = "1.0" encoding = "utf-8" ?>
<hibernate-mapping xmlns = "urn:nhibernate-mapping-2.2" assembly = "NHibernateDemo"
namespace = "NHibernateDemo">
<class name = "Order" table = "`Order`">
<id name = "Id">
<generator class = "guid.comb"/>
</id>
<property name = "Ordered"/>
<property name = "Shipped"/>
<component name = "ShipTo">
<property name = "Street"/>
<property name = "City"/>
<property name = "Province"/>
<property name = "Country"/>
</component>
<many-to-one name = "Customer" column = "CustomerId" cascade = "save-update"/>
</class>
</hibernate-mapping>
Donc, si nous créons une nouvelle commande et qu'un nouveau client y est associé et que nous disons, enregistrez cette commande, nous pourrions vouloir la mettre en cascade. Mais une chose que nous ne voudrions probablement pas faire est si une commande est supprimée pour supprimer le client correspondant.
Donc, ici, nous voudrions faire une mise à jour de sauvegarde, donc en utilisant une mise à jour de sauvegarde, elle mettra en cascade toutes les sauvegardes ou mises à jour à ce client. Donc, si nous avons un nouveau client ou si nous changeons de client, cela se répercutera en cascade. S'il s'agit d'une suppression, cela ne le supprimera pas de la base de données.
Donc, en exécutant à nouveau notre application, tout fonctionne toujours comme prévu.
New Customer:
John Doe (00000000-0000-0000-0000-000000000000)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Unspecified)
CreditRating: Good
AverageRating: 42.42424242
Orders:
Id: 00000000-0000-0000-0000-000000000000
Order Id: 00000000-0000-0000-0000-000000000000
Reloaded:
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
The orders were ordered by:
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
John Doe (10b2a3d7-7fcf-483c-b1da-a5bb00b8512e)
Points: 100
HasGoldStatus: True
MemberSince: 1/1/2012 12:00:00 AM (Utc)
CreditRating: Good
AverageRating: 42.4242
Orders:
Order Id: e6680e30-5b3b-4efa-b017-a5bb00b85133
Order Id: b03858e7-8c36-4555-8878-a5bb00b85134
Press <ENTER> to exit...
Maintenant, vous devriez jeter un œil à votre application, rappelez-vous que la valeur par défaut est Aucune et que vous devez penser à vos entités et à vos relations entre elles pour déterminer les cascades appropriées pour chacune de vos entités ainsi que chacune de vos relations dans cette base de données.