Intégration de System.Transactions à SQL Server
S'applique à : .NET Framework .NET .NET Standard
.NET comprend une infrastructure de transactions à laquelle il est possible d’accéder via l’espace de noms System.Transactions. Cette infrastructure expose les transactions de façon totalement intégrée à .NET, dont ADO.NET.
En plus des améliorations de la programmabilité, System.Transactions et ADO.NET peuvent fonctionner ensemble pour coordonner les optimisations quand des transactions sont utilisées. Une transaction susceptible d'être promue est une transaction légère (locale) qui peut être promue automatiquement en une transaction entièrement distribuée en fonction des besoins.
Le fournisseur de données Microsoft SqlClient pour SQL Server prend en charge les transactions pouvant être promues quand SQL Server est utilisé. Une transaction pouvant être promue n'invoque pas la charge supplémentaire d'une transaction distribuée à moins qu'elle ne soit requise. Les transactions pouvant être promues sont automatiques et ne nécessitent aucune intervention de la part du développeur.
Création de transactions pouvant être promues
Le fournisseur de données Microsoft SqlClient pour SQL Server assure la prise en charge des transactions pouvant être promues, qui sont gérées au moyen de classes dans l’espace de noms System.Transactions. Les transactions pouvant être promues optimisent les transactions distribuées en différant la création d'une transaction distribuée jusqu'à ce qu'elle soit nécessaire. Si un seul gestionnaire de ressources est requis, aucune transaction distribuée n'a lieu.
Notes
Dans un scénario de niveau de confiance partiel, l'objet DistributedTransactionPermission est requis lorsqu'une transaction est promue en transaction distribuée.
Scénarios de transactions pouvant être promues
Les transactions distribuées consomment généralement une partie importante des ressources système, car elles sont gérées par Microsoft Distributed Transaction Coordinator (MS DTC), qui intègre tous les gestionnaires de ressources auxquels la transaction accède. Une transaction pouvant être promue est une forme spéciale de transaction System.Transactions qui délègue efficacement le travail à une simple transaction SQL Server. System.Transactions, Microsoft.Data.SqlClient et SQL Server coordonnent le travail impliqué dans la gestion de la transaction en la promouvant en transaction distribuée complète, si nécessaire.
L'avantage de l'utilisation de transactions pouvant être promues réside dans le fait que, quand une connexion est ouverte à l'aide d'une transaction TransactionScope active alors qu'aucune autre connexion n'est ouverte, la transaction est validée comme transaction légère, au lieu de générer la charge supplémentaire d'une transaction entièrement distribuée.
Mots clés de chaîne de connexion
La propriété ConnectionString prend en charge un mot clé, Enlist
, qui indique si Microsoft.Data.SqlClient détecte des contextes transactionnels et inscrit automatiquement la connexion dans une transaction distribuée. Si Enlist=true
, la connexion est automatiquement inscrite dans le contexte de transaction actuel du thread qui s'ouvre. Si Enlist=false
, la connexion SqlClient
n'interagit pas avec une transaction distribuée. La valeur par défaut de Enlist
est true. Si Enlist
n'est pas spécifié dans la chaîne de connexion, la connexion est automatiquement inscrite dans une transaction distribuée détectée lors de l'ouverture de la connexion.
Les mots clés Transaction Binding
d'une chaîne de connexion SqlConnection contrôlent l'association de la connexion à une transaction System.Transactions
inscrite. Il est également possible d'utiliser la propriété TransactionBinding d'un objet SqlConnectionStringBuilder.
Le tableau suivant décrit les valeurs possibles.
Mot clé | Description |
---|---|
Implicit Unbind | Valeur par défaut. La connexion se détache de la transaction une fois cette dernière terminée et revient au mode de validation automatique. |
Explicit Unbind | La connexion reste attachée à la transaction jusqu'à la fermeture de cette dernière. La connexion échoue si la transaction associée n'est pas active ou ne correspond pas à la propriété Current. |
Utilisation de TransactionScope
La classe TransactionScope crée un bloc de code transactionnel en inscrivant implicitement les connexions dans une transaction distribuée. Vous devez appeler la méthode Complete à la fin du bloc TransactionScope avant de le quitter. Le fait de quitter le bloc invoque la méthode Dispose . Si une exception a été levée qui a pour effet que le code sorte de la portée, la transaction est considérée comme abandonnée.
Il est recommandé d'utiliser un bloc using
pour s'assurer que la méthode Dispose est appelée sur l'objet TransactionScope en cas de sortie du bloc using. La non-validation ou la non-restauration des transactions en attente peut considérablement nuire aux performances, le délai d'attente par défaut de l'objet TransactionScope étant d'une minute. Si vous n'utilisez pas une instruction using
, vous devez effectuer tout le travail dans un bloc Try
et appeler explicitement la méthode Dispose dans le bloc Finally
.
Si une exception est levée dans l'objet TransactionScope, la transaction est marquée comme incohérente et est abandonnée. Elle sera annulée lors de la suppression du TransactionScope . Si aucune exception ne se produit, les transactions sont validées.
Notes
La classe TransactionScope
crée par défaut une transaction avec un IsolationLevel ayant la valeur Serializable
. Selon votre application, vous pouvez envisager de baisser le niveau d'isolation pour éviter une contention élevée au sein de votre application.
Notes
Il est recommandé de n'effectuer des mises à jour, des insertions et des suppressions que dans des transactions distribuées car ces opérations consomment des ressources de base de données importantes. Les instructions select risquent de verrouiller inutilement les ressources de base de données ; dans certains cas, vous pouvez être amené à utiliser des transactions pour effectuer des sélections. Tout travail autre qu'un travail de base de données doit être réalisé en dehors de la transaction, à moins qu'il n'implique d'autres gestionnaires de ressources. Bien qu'une exception dans la transaction empêche la validation de celle-ci, la classe TransactionScope n'a aucune disposition pour annuler les modifications que votre code a apportées en dehors de la transaction proprement dite. Pour intervenir lors de l'annulation de la transaction, vous devez écrire votre propre implémentation de l'interface IEnlistmentNotification et vous inscrire explicitement dans la transaction.
Exemple
L'utilisation de l'espace de noms System.Transactions exige que vous disposiez d'une référence à System.Transactions.dll.
La fonction suivante montre comment créer une transaction pouvant être promue en relation avec deux instances différentes de SQL Server, représentées par deux objets SqlConnection différents, enveloppés dans un bloc TransactionScope .
Le code ci-dessous crée le bloc TransactionScope avec une instruction using
et ouvre la première connexion, qui l’inscrit automatiquement dans TransactionScope.
La transaction est inscrite initialement comme transaction légère, et non comme transaction distribuée complète. La seconde connexion est inscrite dans l'objet TransactionScope uniquement si la commande dans la première connexion ne lève pas une exception. Une fois la seconde connexion ouverte, la transaction est automatiquement promue en transaction entièrement distribuée.
Par la suite, la méthode Complete est appelée, qui ne valide la transaction que si aucune exception n’a été levée. Si une exception a été levée dans le bloc TransactionScope , la méthode Complete
n'est pas appelée et la transaction distribuée est annulée lors de la suppression de l'objet TransactionScope à la fin de son bloc using
.
using System;
using System.Transactions;
using Microsoft.Data.SqlClient;
class Program
{
static void Main(string[] args)
{
string connectionString = "Data Source = localhost; Integrated Security = true; Initial Catalog = AdventureWorks";
string commandText1 = "INSERT INTO Production.ScrapReason(Name) VALUES('Wrong size')";
string commandText2 = "INSERT INTO Production.ScrapReason(Name) VALUES('Wrong color')";
int result = CreateTransactionScope(connectionString, connectionString, commandText1, commandText2);
Console.WriteLine("result = " + result);
}
static public int CreateTransactionScope(string connectString1, string connectString2,
string commandText1, string commandText2)
{
// Initialize the return value to zero and create a StringWriter to display results.
int returnValue = 0;
System.IO.StringWriter writer = new System.IO.StringWriter();
// Create the TransactionScope in which to execute the commands, guaranteeing
// that both commands will commit or roll back as a single unit of work.
using (TransactionScope scope = new TransactionScope())
{
using (SqlConnection connection1 = new SqlConnection(connectString1))
{
try
{
// Opening the connection automatically enlists it in the
// TransactionScope as a lightweight transaction.
connection1.Open();
// Create the SqlCommand object and execute the first command.
SqlCommand command1 = new SqlCommand(commandText1, connection1);
returnValue = command1.ExecuteNonQuery();
writer.WriteLine("Rows to be affected by command1: {0}", returnValue);
// if you get here, this means that command1 succeeded. By nesting
// the using block for connection2 inside that of connection1, you
// conserve server and network resources by opening connection2
// only when there is a chance that the transaction can commit.
using (SqlConnection connection2 = new SqlConnection(connectString2))
try
{
// The transaction is promoted to a full distributed
// transaction when connection2 is opened.
connection2.Open();
// Execute the second command in the second database.
returnValue = 0;
SqlCommand command2 = new SqlCommand(commandText2, connection2);
returnValue = command2.ExecuteNonQuery();
writer.WriteLine("Rows to be affected by command2: {0}", returnValue);
}
catch (Exception ex)
{
// Display information that command2 failed.
writer.WriteLine("returnValue for command2: {0}", returnValue);
writer.WriteLine("Exception Message2: {0}", ex.Message);
}
}
catch (Exception ex)
{
// Display information that command1 failed.
writer.WriteLine("returnValue for command1: {0}", returnValue);
writer.WriteLine("Exception Message1: {0}", ex.Message);
}
}
// If an exception has been thrown, Complete will not
// be called and the transaction is rolled back.
scope.Complete();
}
// The returnValue is greater than 0 if the transaction committed.
if (returnValue > 0)
{
writer.WriteLine("Transaction was committed.");
}
else
{
// You could write additional business logic here, notify the caller by
// throwing a TransactionAbortedException, or log the failure.
writer.WriteLine("Transaction rolled back.");
}
// Display messages.
Console.WriteLine(writer.ToString());
return returnValue;
}
}