Partager via


Réplication de colonnes d'identité

Quand vous attribuez la propriété IDENTITY à une colonne, MicrosoftSQL Server génère automatiquement des numéros séquentiels pour les nouvelles lignes insérées dans la table contenant la colonne d'identité. Pour plus d'informations, consultez IDENTITY (Propriété) (Transact-SQL). Les colonnes d'identité devant être incluses comme composantes de la clé primaire, il est important d'éviter les valeurs dupliquées dans les colonnes d'identité. Pour utiliser des colonnes d'identité dans une topologie de réplication qui a des mises à jour sur plusieurs nœuds, chaque nœud de cette topologie de réplication doit avoir une plage différente de valeurs d'identité, de façon à ce qu'il n'y ait pas de valeurs dupliquées.

Par exemple, le serveur de publication peut avoir la plage 1 à 100, l'Abonné A 101 à 200 et l'Abonné B 201 à 300. Si une ligne est insérée sur le serveur de publication et que la valeur d'identité est de 65, cette valeur est répliquée vers chaque Abonné. Quand la réplication insère des données sur chaque Abonné, elle n'incrémente pas la valeur de la colonne d'identité dans la table de l'Abonné ; c'est au contraire la valeur littérale 65 qui est insérée. Seules les insertions des utilisateurs - et pas les insertions des agents de réplication - provoquent l'incrémentation de la valeur de la colonne d'identité.

La réplication gère les colonnes d'identité à travers tous les types de publications et d'abonnements ; elle vous permet de gérer les colonnes manuellement ou de demander à ce que la réplication les gère automatiquement.

[!REMARQUE]

L'ajout d'une colonne d'identité à une table publiée n'est pas prise en charge, car elle peut aboutir à une non-convergence quand la colonne est répliquée vers l'Abonné. Les valeurs de la colonne d'identité sur l'Abonné dépendent de l'ordre dans lequel les lignes sont physiquement stockées dans la table affectée. Les lignes sont susceptibles d'être stockées différemment sur l'Abonné ; la valeur de la colonne d'identité peut donc être différente pour les mêmes lignes.

Spécification d'une option de gestion des plages d'identités

La réplication offre trois options de gestion des plages d'identités :

  • Automatique. Utilisée pour la réplication de fusion et la réplication transactionnelle avec des mises à jour sur l'Abonné. Spécifiez des plages de taille pour le serveur de publication et les Abonnés, et la réplication gère automatiquement l'affectation des nouvelles plages. La réplication définit l'option NOT FOR REPLICATION pour la colonne d'identité sur l'Abonné, de sorte que seules les insertions des utilisateurs provoquent l'incrémentation de la valeur sur l'Abonné. Pour plus d'informations, consultez Contrôle des contraintes, des identités et des déclencheurs avec l'option NOT FOR REPLICATION.

    [!REMARQUE]

    Les Abonnés doivent se synchroniser avec le serveur de publication pour recevoir de nouvelles plages. Les plages d'identité étant affectées automatiquement aux Abonnés, il est possible pour les Abonnés d'épuiser la totalité des plages d'identité disponibles s'ils requièrent de nouvelles plages de façon répétée.

  • Manuel. Utilisée pour la réplication de fusion et la réplication transactionnelle sans mises à jour sur l'Abonné, pour la réplication transactionnelle d'égal à égal ou si votre application doit contrôler les plages d'identité par programmation. Si vous spécifiez une gestion manuelle, vous devez vérifier que les plages sont affectées au serveur de publication et à chaque Abonné, et que de nouvelles plages sont attribuées si les plages initiales sont utilisées. La réplication définit l'option NOT FOR REPLICATION sur la colonne d'identité sur l'Abonné.

  • Aucune. Cette option n'est recommandée que pour assurer une compatibilité amont avec les versions antérieures de SQL Server et elle est disponible seulement à partir de l'interface des procédures stockées pour les publications transactionnelles.

Pour spécifier une option de gestion des plages d'identité

Attribution de plages d'identité

La réplication de fusion et la réplication transactionnelle utilisent différentes méthodes pour l'attribution des plages ; ces méthodes sont décrites dans cette section.

Il y a deux types de plages à prendre en compte lors de la réplication de colonnes d'identité : les plages attribuées au serveur de publication et aux Abonnés, et la plage du type de données de la colonne. Le tableau suivant indique les plages disponibles pour les types de données généralement utilisés dans les colonnes d'identité. La plage est utilisée à travers tous les nœuds d'une topologie. Par exemple, si vous utilisez smallint en démarrant à 1 avec un incrément de 1, le nombre maximal d'insertions est de 32 767 pour le serveur de publication et pour tous les Abonnés. Le nombre réel d'insertions dépend de la présence d'interruptions dans les valeurs et de l'utilisation d'une valeur de seuil. Pour plus d'informations sur les seuils, consultez les sections suivantes, « Réplication de fusion » et « Réplication transactionnelle avec des abonnements mis à jour en attente ».

Si le serveur de publication épuise sa plage d'identité après une insertion, il peut automatiquement attribuer une nouvelle plage si l'insertion a été effectuée par un membre du rôle de base de données fixe db_owner. Si l'insertion a été effectuée par un utilisateur qui ne fait pas partie de ce rôle, l'Agent de lecture du journal, l'Agent de fusion ou un utilisateur qui est membre du rôle db_owner doit exécuter sp_adjustpublisheridentityrange (Transact-SQL). Pour les publications transactionnelles, l'Agent de lecture du journal doit être en cours d'exécution pour allouer automatiquement une nouvelle plage (la configuration par défaut est que l'agent s'exécute en continu).

Type de données

Plage

tinyint

Non pris en charge pour la gestion automatique

smallint

-2^15 (-32 768) à 2^15-1 (32 767)

int

-2^31 (-2 147 483 648) à 2^31-1 (2 147 483 647)

bigint

-2^63 (-9 223 372 036 854 775 808) à 2^63-1 (9 223 372 036 854 775 807)

decimal et numeric

-10^38+1 à 10^38-1

Réplication de fusion

Les plages d'identité sont gérées par le serveur de publication et propagées aux Abonnés par l'Agent de fusion (dans une hiérarchie de réédition, les plages sont gérées par le serveur de publication racine et les rééditeurs). Les valeurs d'identité sont affectées à partir d'un pool au serveur de publication. Quand vous ajoutez un article avec une colonne d'identité à une publication dans l'Assistant Nouvelle publication ou à l'aide de sp_addmergearticle (Transact-SQL), vous spécifiez des valeurs pour :

  • Le paramètre @identity_range, qui contrôle la taille de la plage d'identités initialement allouée à la fois au serveur de publication et aux Abonnés avec des abonnements clients.

    [!REMARQUE]

    Pour les Abonnés exécutant des versions antérieures de SQL Server, ce paramètre (et non pas le paramètre @pub_identity_range) contrôle également la taille de la plage d'identité sur les Abonnés de réédition.

  • Le paramètre @pub_identity_range, qui contrôle la taille de la plage d'identités pour la réédition allouée aux Abonnés avec des abonnements serveur (requis pour les données de réédition). Tous les Abonnés avec des abonnements serveur reçoivent une plage pour la réédition, même s'ils ne rééditent pas réellement des données.

  • Le paramètre @threshold, qui est utilisé pour déterminer quand une nouvelle plage d'identités est requise pour un abonnement à SQL Server Compact 3.5 SP1 ou à une version antérieure de SQL Server.

Par exemple, vous pouvez spécifier 10 000 pour @identity_range et 500 000 pour @pub_identity_range. Une plage principale de 10 000 est assignée au serveur de publication et à tous les Abonnés qui exécutent SQL Server 2005 ou une version ultérieure, y compris l'Abonné avec l'abonnement serveur. L'Abonné avec l'abonnement serveur se voit également attribuer une plage principale de 500 000, qui peut être utilisée par les Abonnés qui se synchronisent avec l'Abonné de republication (vous devez également spécifier @identity_range, @pub_identity_range et @threshold pour les articles de la publication sur l'Abonné de republication).

Chaque Abonné exécutant SQL Server 2005 ou une version ultérieure reçoit aussi une plage d'identités secondaire. La plage secondaire a une taille équivalente à celle de la plage principale ; quand la plage principale est épuisée, la plage secondaire est utilisée et l'Agent de fusion attribue une nouvelle plage à l'Abonné. La nouvelle plage devient la plage secondaire et le processus continue tant que l'Abonné utilise des valeurs d'identité.

Les Abonnés exécutant SQL Server Compact 3.5 SP1 ou des versions antérieures de SQL Server se voient attribuer seulement la plage principale ; l'attribution de nouvelles plages est contrôlée par le paramètre @threshold. Un Abonné de réédition a seulement la plage spécifiée dans le paramètre @identity_range ; il doit utiliser cette plage pour les modifications locales et pour les modifications sur les Abonnés qui se synchronisent avec l'Abonné de réédition. Par exemple, vous pouvez spécifier 10 000 pour @pub_identity_range, 500 000 pour @identity_range et 80 pour cent pour @threshold. Après 8 000 insertions sur un Abonné (80 pour cent de 10 000), le serveur de publication se voit attribuer une nouvelle plage. Quand une nouvelle plage est attribuée, il y a une interruption dans les valeurs de plage d'identités de la table. La spécification d'un seuil plus élevé aboutit à des interruptions plus courtes, mais le système présente alors une tolérance de pannes moindre : si l'Agent de fusion ne peut pas s'exécuter pour une raison quelconque, un Abonné peut tomber plus facilement à court d'identités.

Réplication transactionnelle avec des abonnements mis à jour en attente

Les plages d'identités sont gérées par le serveur de distribution et propagées aux Abonnés par l'Agent de distribution. Les valeurs d'identité sont attribuées à partir d'un pool au serveur de distribution. La taille du pool est basée sur la taille du type de données et sur l'incrément utilisé par la colonne d'identité. Quand vous ajoutez un article avec une colonne d'identité à une publication dans l'Assistant Nouvelle publication ou à l'aide de sp_addarticle (Transact-SQL), vous spécifiez des valeurs pour :

  • Le paramètre @identity_range, qui contrôle la taille de la plage d'identité initialement allouée à tous les Abonnés.

  • Le paramètre @pub_identity_range, qui contrôle la taille de la plage d'identité allouée au serveur de publication.

  • Le paramètre @threshold, qui est utilisé pour déterminer quand une nouvelle plage d'identités est requise pour un abonnement.

Par exemple, vous pouvez spécifier 10 000 pour @pub_identity_range, 1 000 pour @identity_range (en faisant l'hypothèse d'un nombre moins élevé de mises à jour sur l'Abonné) et 80 pour cent pour @threshold. Après 800 insertions sur un Abonné (80 pour cent de 1 000), un Abonné se voit attribuer une nouvelle plage. Après 8 000 insertions sur l'Abonné, le serveur de publication se voit attribuer une nouvelle plage. Quand une nouvelle plage est attribuée, il y aura une interruption dans les valeurs de plage d'identités de la table. La spécification d'un seuil plus élevé aboutit à des interruptions plus courtes, mais le système présente alors une tolérance de pannes moindre : si l'Agent de distribution ne peut pas s'exécuter pour une raison quelconque, un Abonné peut tomber plus facilement à court d'identités.

Attribution de plages pour la gestion manuelle des plages d'identité

Si vous spécifiez une gestion manuelle des plages d'identité, vous devez vérifier que le serveur de publication et que chaque Abonné utilisent des plages d'identités différentes. Par exemple, supposons une table sur le serveur de publication avec une colonne d'identité définie en IDENTITY(1,1) : la colonne d'identité commence à 1 et elle est incrémentée de 1 chaque fois qu'une ligne est insérée. Si la table sur le serveur de publication a 5 000 lignes et que vous vous attendez à un accroissement de la table au cours de la durée de vie de l'application, le serveur de publication peut utiliser la plage 1 à 10 000. Étant donnés deux Abonnés, l'Abonné A peut utiliser la plage 10 001 à 20 000 et l'Abonné B peut utiliser la plage 20 001 à 30 000.

Après qu'un Abonné ait été initialisé avec une capture instantanée ou via un autre moyen, exécutez DBCC CHECKIDENT pour attribuer un point de départ pour sa plage d'identités. Par exemple, sur l'Abonné A, vous pouvez exécuter DBCC CHECKIDENT('<TableName>','reseed',10001). Sur l'Abonné B, vous pouvez exécuter CHECKIDENT('<TableName>','reseed',20001).

Pour attribuer de nouvelles plages au serveur de publication ou aux Abonnés, exécutez DBCC CHECKIDENT et spécifiez une nouvelle valeur permettant de réalimenter la table. Vous devez disposer d'un moyen quelconque pour déterminer quand une nouvelle plage doit être attribuée. Par exemple, votre application peut avoir un mécanisme détectant quand un nœud est sur le point d'utiliser la totalité de sa plage et attribuant une nouvelle plage à l'aide de DBCC CHECKIDENT. Vous pouvez aussi ajouter un contrainte de vérification pour garantir qu'une ligne ne peut pas être ajoutée si une valeur de colonne d'identité en dehors de la page devait être utilisée.

Gestion des plages d'identités après la restauration d'une base de données

Si vous utilisez la gestion automatique des plages d'identités, quand un Abonné est restauré à partir d'une sauvegarde, il requiert automatiquement une nouvelle plage de valeurs d'identité. Si un serveur de publication est restauré à partir d'une sauvegarde, vous devez vérifier que le serveur de publication se voit attribuer une plage appropriée. Pour la réplication de fusion, attribuez une nouvelle plage à l'aide de sp_restoremergeidentityrange (Transact-SQL). Pour la réplication transactionnelle, déterminez la valeur la plus élevée qui a été utilisée, puis définissez le point de départ pour les nouvelles plages. Utilisez la procédure suivante après que la base de données de publication ait été restaurée :

  1. Arrêtez toutes les activités sur tous les Abonnés.

  2. Pour chaque table publiée comportant une colonne d'identité :

    1. Dans la base de données d'abonnement sur chaque Abonné, exécutez IDENT_CURRENT('<TableName>').

    2. Enregistrez la valeur la plus élevée trouvée parmi tous les Abonnés.

    3. Dans la base de données de publication sur le serveur de publication, exécutez DBCC CHECKIDENT(<TableName>','reseed',<HighestValueFound+1>.

    4. Dans la base de données de publication sur le serveur de publication, exécutez sp_adjustpublisheridentityrange <PublicationName>, <TableName>.

    [!REMARQUE]

    Si la valeur de la colonne d'identité est définie pour se décrémenter au lieu de s'incrémenter, enregistrez la valeur la plus faible, puis réalimentez la table avec cette valeur.