Fonctions scalaires CLR
S'applique à :SQL Server
Une fonction scalaire (SVF) retourne une valeur unique, telle qu’une chaîne, un entier ou une valeur de bits. Vous pouvez créer des fonctions scalaires définies par l’utilisateur dans du code managé à l’aide de n’importe quel langage de programmation .NET Framework. Ces fonctions sont accessibles à Transact-SQL ou à d’autres codes managés. Pour plus d’informations sur les avantages de l’intégration clR (Common Language Runtime) et le choix entre le code managé et Transact-SQL, consultez vue d’ensemble de l’intégration CLR.
Conditions requises pour les fonctions scalaires CLR
Les fonctions scalaires .NET Framework sont implémentées en tant que méthodes d'une classe dans un assembly .NET Framework. Les paramètres d’entrée et le type retournés à partir d’une fonction SVF peuvent être l’un des types de données scalaires pris en charge par SQL Server, sauf varchar, char, rowversion, de texte, ntext, image, timestamp, de table ou curseur. Les fichiers SVFs doivent garantir une correspondance entre le type de données SQL Server et le type de données de retour de la méthode d’implémentation. Pour plus d’informations sur les conversions de types, consultez mapper les données de paramètres CLR.
Lorsque vous implémentez un .NET Framework SVF dans un langage .NET Framework, vous pouvez spécifier l’attribut personnalisé SqlFunction
pour inclure des informations supplémentaires sur la fonction. L’attribut SqlFunction
indique si la fonction accède ou non aux données, si elle est déterministe et si la fonction implique des opérations à virgule flottante.
Les fonctions scalaires définies par l’utilisateur peuvent être déterministes ou non déterministes. Une fonction déterministe retourne toujours le même résultat lorsqu’elle est appelée avec un ensemble spécifique de paramètres d’entrée. Une fonction non déterministe peut retourner des résultats différents lorsqu’elle est appelée avec un ensemble spécifique de paramètres d’entrée.
Remarque
Ne marquez pas une fonction comme déterministe si la fonction ne produit pas toujours les mêmes valeurs de sortie, étant donné les mêmes valeurs d’entrée et le même état de base de données. Le marquage d'une fonction comme étant déterministe alors que cette dernière ne l'est pas vraiment peut provoquer une altération des vues indexées et des colonnes calculées. Vous marquez une fonction comme déterministe en définissant la propriété IsDeterministic
sur true.
Paramètres table
Les paramètres table (types de tables définis par l'utilisateur et passés dans une procédure ou une fonction) offrent un moyen efficace pour passer plusieurs lignes de données au serveur. Les tvps offrent des fonctionnalités similaires aux tableaux de paramètres, mais offrent une plus grande flexibilité et une intégration plus étroite avec Transact-SQL. Ils sont également susceptibles de générer de meilleures performances.
Les paramètres table aident également à réduire le nombre d'allers-retours au serveur. Au lieu d'envoyer plusieurs demandes au serveur, comme avec une liste de paramètres scalaires, les données peuvent être envoyées au serveur en tant que paramètres table. Un type de table défini par l’utilisateur ne peut pas être transmis en tant que paramètre table à une procédure stockée managée ou à une fonction exécutée dans le processus SQL Server. Pour plus d’informations sur les tvps, consultez Utiliser des paramètres table (moteur de base de données).
Exemple de fonction scalaire CLR
Voici un svF simple qui accède aux données et retourne une valeur entière :
using Microsoft.SqlServer.Server;
using System.Data.SqlClient;
public class T
{
[SqlFunction(DataAccess = DataAccessKind.Read)]
public static int ReturnOrderCount()
{
using (SqlConnection conn
= new SqlConnection("context connection=true"))
{
conn.Open();
SqlCommand cmd = new SqlCommand(
"SELECT COUNT(*) AS 'Order Count' FROM SalesOrderHeader", conn);
return (int)cmd.ExecuteScalar();
}
}
}
La première ligne de références de code Microsoft.SqlServer.Server
pour accéder aux attributs et System.Data.SqlClient
pour accéder à l’espace de noms ADO.NET. (Cet espace de noms contient SqlClient
, le fournisseur de données .NET Framework pour SQL Server.)
Ensuite, la fonction reçoit l’attribut personnalisé SqlFunction
, qui se trouve dans l’espace de noms Microsoft.SqlServer.Server
. L'attribut personnalisé indique si la fonction définie par l'utilisateur utilise le fournisseur in-process pour lire les données sur le serveur. SQL Server n’autorise pas les fonctions définies par l’utilisateur à mettre à jour, insérer ou supprimer des données. SQL Server peut optimiser l’exécution d’une fonction UDF qui n’utilise pas le fournisseur in-process. Cela est indiqué en définissant DataAccessKind
sur DataAccessKind.None
. Sur la ligne suivante, la méthode cible est une méthode statique publique (partagée dans Visual Basic .NET).
La classe SqlContext
, située dans l’espace de noms Microsoft.SqlServer.Server
, peut ensuite accéder à un objet SqlCommand
avec une connexion à l’instance SQL Server déjà configurée. Bien qu’il ne soit pas utilisé ici, le contexte de transaction actuel est également disponible via l’interface de programmation d’application (API) System.Transactions
.
La plupart des lignes de code dans le corps de la fonction doivent être familières aux développeurs qui écrivent des applications clientes qui utilisent les types trouvés dans l’espace de noms System.Data.SqlClient
.
using(SqlConnection conn = new SqlConnection("context connection=true"))
{
conn.Open();
SqlCommand cmd = new SqlCommand(
"SELECT COUNT(*) AS 'Order Count' FROM SalesOrderHeader", conn);
return (int) cmd.ExecuteScalar();
}
Le texte de commande approprié est spécifié en initialisant l’objet SqlCommand
. L’exemple précédent compte le nombre de lignes de la table SalesOrderHeader
. Ensuite, la méthode ExecuteScalar
de l’objet cmd
est appelée. Elle retourne une valeur de type int basée sur la requête. Pour finir, le nombre de commandes (« order count ») est retourné à l'appelant.
Si ce code est enregistré dans un fichier nommé FirstUdf.cs, il peut être compilé dans un assembly comme suit :
/t:library
indique qu'une bibliothèque, plutôt qu'un fichier exécutable, doit être produite. Les exécutables ne peuvent pas être inscrits dans SQL Server.
Les objets de base de données Visual C++ compilés avec /clr:pure
ne sont pas pris en charge pour l’exécution sur SQL Server. Il s'agit par exemple d'objets de base de données tels que des fonctions scalaires.
La requête Transact-SQL et un exemple d’appel pour inscrire l’assembly et la fonction UDF sont les suivants :
CREATE ASSEMBLY FirstUdf
FROM 'FirstUdf.dll';
GO
CREATE FUNCTION CountSalesOrderHeader()
RETURNS INT
AS EXTERNAL NAME FirstUdf.T.ReturnOrderCount;
GO
SELECT dbo.CountSalesOrderHeader();
GO
Le nom de la fonction tel qu’exposé dans Transact-SQL n’a pas besoin de correspondre au nom de la méthode statique publique cible.