Spécification d'un test de nœud dans une étape d'expression du chemin d'accès
Une étape d'axe dans une expression de chemin d'accès inclut les composants suivants :
Test de nœud
Pour plus d'informations, voir Expressions de chemin d'accès (XQuery).
Un test de nœud est une condition et représente le deuxième composant de l'étape d'axe dans une expression de chemin d'accès. Tous les nœuds sélectionnés par une étape doivent satisfaire cette condition. Dans le cas de l'expression du chemin d'accès /child::ProductDescription, le test de nœud correspond donc à ProductDescription. En d'autres termes, cette étape ne récupère que les enfants du nœud élément dont le nom est ProductDescription.
Une condition du test de nœud peut inclure les éléments suivants :
Nom de nœud. Seuls les nœuds de type identique au nœud principal et portant le nom indiqué sont renvoyés.
Type de nœud. Seuls les nœuds du type précisé sont renvoyés.
[!REMARQUE]
Les noms de nœuds précisés dans les expressions de chemin d'accès XQuery ne sont pas soumis aux mêmes règles dépendantes du classement que les requêtes Transact-SQL et respectent toujours la casse.
Nom de nœud en tant que test de nœud
Pour indiquer un nom de nœud en tant que test de nœud dans une étape d'expression du chemin d'accès, vous devez au préalable comprendre le concept de type de nœud principal. Chaque axe child (enfant), parent ou attribute (attribute) possède un type de nœud principal. Par exemple :
Un axe attribute ne peut contenir que des attributs. L'axe attribute correspond par conséquent au type de nœud principal de l'axe.
Pour les autres axes, si les nœuds sélectionnés par l'axe peuvent contenir des nœuds éléments, l'élément correspond alors au type de nœud principal de cet axe.
Si vous indiquez un nom de nœud en tant que test de nœud, l'étape renvoie les types de nœuds suivants :
Nœuds du type de nœud principal de l'axe
Nœuds portant le même nom que celui indiqué dans le test de nœud.
Par exemple, considérons l'expression de chemin d'accès suivante :
child::ProductDescription
Cette expression à une seule étape indique un axe enfant (child) et précise le nom de nœud ProductDescription comme test de nœud. L'expression ne renvoie que les nœuds du type du nœud principal de l'axe enfant, les nœuds éléments, et dont le nom est ProductDescription.
L'expression du chemin d'accès /child::PD:ProductDescription/child::PD:Features/descendant::*, possède pour sa part trois étapes. Ces étapes indiquent les axes child et descendant. Dans chacune des étapes, le nom du nœud est spécifié comme test de nœud. Le caractère générique (à savoir *) de la troisième étape indique tous les nœuds du type de nœud principal correspondant à l'axe descendant. Le type de nœud principal de l'axe détermine en effet le type de nœuds sélectionné, ainsi que les filtres de nom de nœud que les nœuds ont sélectionnés.
Ainsi, si cette expression est exécutée sur des documents XML tirés du catalogue de produits de la table ProductModel, elle récupère tous les enfants des nœuds éléments de l'enfant de nœud élément <Features> de l'élément <ProductDescription>.
L'expression de chemin d'accès /child::PD:ProductDescription/attribute::ProductModelID est composée de deux étapes. Ces deux étapes indiquent un nom de nœud comme test de nœud. La deuxième étape utilise également un axe attribute. Ainsi, chaque étape sélectionne des nœuds du type de nœud principal de son axe portant le nom spécifié comme test de nœud. L'expression renvoie donc le nœud de l'attribut ProductModelID du nœud élément <ProductDescription>.
En spécifiant les noms des nœuds comme tests de nœud, vous pouvez également utiliser le caractère générique (*) dans le nom local d'un nœud ou dans le préfixe de son espace de noms, comme illustré dans l'exemple suivant :
declare @x xml
set @x = '
<greeting xmlns="ns1">
<salutation>hello</salutation>
</greeting>
<greeting xmlns="ns2">
<salutation>welcome</salutation>
</greeting>
<farewell xmlns="ns1" />'
select @x.query('//*:greeting')
select @x.query('declare namespace ns="ns1"; /ns:*')
Type de nœud comme test de nœud
Pour interroger des types de nœuds autres que des nœuds éléments, lancez un test de type de nœud. Le tableau suivant répertorie les quatre tests de type de nœud disponibles.
Type de nœud |
Valeur renvoyée |
Exemple |
---|---|---|
comment() |
Vrai (valeur True) dans le cas d'un nœud de commentaire. |
following::comment() sélectionne tous les nœuds de commentaire apparaissant après le nœud de contexte. |
node() |
Vrai (valeur True) dans le cas d'un nœud de type quelconque. |
preceding::node() sélectionne tous les nœuds apparaissant avant le nœud de contexte. |
processing-instruction() |
Vrai (valeur True) dans le cas d'un nœud d'instructions de traitement. |
self::processing instruction() sélectionne tous les nœuds d'instructions de traitement apparaissant dans le nœud de contexte. |
text() |
Vrai (valeur True) dans le cas d'un nœud de texte. |
child::text() sélectionne tous les nœuds de texte enfants du nœud de contexte. |
Si le type de nœud, tel que text() ou comment(), est spécifié comme test de nœud, l'étape ne renvoie que les nœuds du type spécifié indépendamment du type de nœud principal de l'axe. Par exemple, l'expression de chemin d'accès suivant ne renvoie que les enfants du nœud de commentaire du nœud de contexte :
child::comment()
De même, /child::ProductDescription/child::Features/child::comment() récupère les enfants du nœud de commentaire provenant de l'enfant du nœud éléments <Features>, lui même provenant du nœud élément <ProductDescription>.
Exemples
Les exemples suivants comparent nom de nœud et type de nœud.
A.Résultats de la spécification du nom de nœud et du type de nœud en tant que tests de nœuds dans une expression de chemin d'accès
L'exemple suivant propose un document XML simple affecté à une variable de type xml. Le document est interrogé à l'aide de plusieurs expressions de chemin d'accès. Les résultats sont ensuite comparés.
declare @x xml
set @x='
<a>
<b>text1
<c>text2
<d>text3</d>
</c>
</b>
</a>'
select @x.query('
/child::a/child::b/descendant::*
')
Cette expression recherche les nœuds éléments descendants du nœud élément <b>.
L'astérisque (*) figurant dans le test de nœud indique un caractère générique représentant le nom du nœud. L'axe descendant possède un type de nœud primaire défini sur le nœud élément. Ainsi, l'expression renvoie tous les nœuds éléments descendants du nœud élément <b>. En d'autres termes, les nœuds éléments <c> et <d> sont renvoyés, comme illustré dans le résultat ci-dessous :
<c>text2
<d>text3</d>
</c>
<d>text3</d>
Si vous indiquez un axe descendant-or-self plutôt qu'un axe descendant, le nœud de contexte est renvoyé tout comme ses descendants :
/child::a/child::b/descendant-or-self::*
Cette expression renvoie le nœud élément <b> ainsi que les nœuds éléments descendants. En renvoyant les nœuds descendants, le type de nœud primaire de l'axe descendant-or-self, correspondant au type de nœud élément, détermine les type de nœuds qui sont renvoyés.
Voici le résultat obtenu :
<b>text1
<c>text2
<d>text3</d>
</c>
</b>
<c>text2
<d>text3</d>
</c>
<d>text3</d>
L'expression précédente utilisait un caractère générique comme nom de nœud. Vous pouvez à la place utiliser la fonction node() comme illustré dans l'expression suivante :
/child::a/child::b/descendant::node()
Puisque node() est un type de nœud, tous les nœuds de l'axe descendant sont renvoyés. Voici le résultat obtenu :
text1
<c>text2
<d>text3</d>
</c>
text2
<d>text3</d>
text3
Si vous indiquez un axe descendant-or-self et node() comme test de nœud, tous les descendants, éléments et nœuds de texte, ainsi que le nœud de contexte, à savoir l'élément <b>, sont alors renvoyés.
<b>text1
<c>text2
<d>text3</d>
</c>
</b>
text1
<c>text2
<d>text3</d>
</c>
text2
<d>text3</d>
text3
B.Indication d'un nom de nœud dans le test de nœud
Cet exemple spécifie un nom de nœud comme test de nœud dans toutes les expressions de chemin d'accès. Par conséquent, toutes les expressions renvoient les nœuds du type de nœud principal de l'axe dont le nom de nœud est indiqué dans le test de nœud.
L'expression de requête suivante renvoie l'élément <Warranty> du document XML relatif au catalogue de produits stocké dans la table Production.ProductModel :
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::PD:Features/child::wm:Warranty
')
FROM Production.ProductModel
WHERE ProductModelID=19
Notez les points suivants dans la requête précédente :
Le mot clé namespace mentionné dans le prologue XQuery définit un préfixe utilisé dans le corps de la requête. Pour plus d'informations sur le prologue XQuery, consultez Prologue XQuery.
Les trois étapes de l'expression de chemin d'accès indiquent l'axe enfant et un nom de nœud comme test de nœud.
La partie facultative de qualificateur d'étape de l'étape d'axe n'est précisée dans aucune étape de l'expression.
La requête renvoie les enfants de l'élément <Warranty> issus de l'enfant de l'élément <Features> provenant lui même de l'élément <ProductDescription>.
Voici le résultat obtenu :
<wm:Warranty xmlns:wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain">
<wm:WarrantyPeriod>3 years</wm:WarrantyPeriod>
<wm:Description>parts and labor</wm:Description>
</wm:Warranty>
Dans la requête suivante, l'expression de chemin d'accès indique un caractère générique (*) dans un test de nœud.
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::PD:Features/child::*
')
FROM Production.ProductModel
WHERE ProductModelID=19
Le caractère générique est précisé pour le nom du nœud. La requête renvoie ainsi tous les enfants du nœud élément <Features> issu de l'enfant du nœud élément provenant du nœud élément <ProductDescription>.
Cette requête est similaire à la précédente sauf qu'un espace de noms combiné au caractère générique est indiqué. Tous les enfants du nœud élément se trouvant dans cet espace de noms sont ainsi renvoyés. Remarque : l'élément <Features> peut contenir des éléments provenant de plusieurs espaces de noms.
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::PD:Features/child::wm:*
')
FROM Production.ProductModel
WHERE ProductModelID=19
Vous pouvez utiliser le caractère générique en tant que préfixe d'espace de noms, comme illustré dans la requête suivante :
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::PD:Features/child::*:Maintenance
')
FROM Production.ProductModel
WHERE ProductModelID=19
Cette requête renvoie les enfants du nœud élément <Maintenance> dans tous les espaces de noms provenant du document XML du catalogue des produits.
C.Indication du type de nœud dans le test de nœud
Cet exemple spécifie le type du nœud servant de base au test de ce nœud dans toutes les expressions de chemin d'accès. Ainsi, toutes les expressions de chemin d'accès renvoient les nœuds de type indiqué dans le test de nœud.
Dans la requête suivante, l'expression de chemin d'accès précise un type de nœud dans sa troisième étape :
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::PD:Features/child::text()
')
FROM Production.ProductModel
WHERE ProductModelID=19
Dans la requête qui suit, les points ci-dessous sont renseignés :
L'expression du chemin d'accès possède trois étapes séparées par une barre oblique (/).
Chacune de ces étapes indique un axe child (enfant).
Les deux premières étapes précisent un nom de nœud comme test de nœud alors que la troisième indique un type de nœud comme test de nœud.
L'expression renvoie les enfants du nœud de texte issus de l'enfant de l'élément <Features> provenant lui-même du nœud élément <ProductDescription>.
Un seul nœud de texte est renvoyé. Voici le résultat obtenu :
These are the product highlights.
La requête suivante renvoie les enfants du nœud de commentaire de l'élément <ProductDescription> :
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::PD:ProductDescription/child::comment()
')
FROM Production.ProductModel
WHERE ProductModelID=19
Notez les points suivants dans la requête précédente :
La deuxième étape indique un type de nœud comme test de nœud.
La requête renvoie comme résultat les enfants du nœud de commentaire provenant des nœuds éléments <ProductDescription>.
Voici le résultat obtenu :
<!-- add one or more of these elements... one for each specific product in this product model -->
<!-- add any tags in <specifications> -->
La requête qui suit récupère les nœuds d'instructions de traitement de premier niveau :
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::processing-instruction()
')
FROM Production.ProductModel
WHERE ProductModelID=19
Voici le résultat obtenu :
<?xml-stylesheet href="ProductDescription.xsl" type="text/xsl"?>
Vous pouvez transmettre un paramètre littéral de chaîne au test de nœud processing-instruction(). Dans ce cas, la requête renvoie les instructions de traitement dont la valeur d'attribut de nom est le littéral de chaîne indiqué dans l'argument.
SELECT CatalogDescription.query('
declare namespace PD="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelDescription";
declare namespace wm="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelWarrAndMain";
/child::processing-instruction("xml-stylesheet")
')
FROM Production.ProductModel
WHERE ProductModelID=19
Limites de mise en œuvre
Les limites suivantes s'appliquent :
Les tests de nœud SequenceType étendu ne sont pas pris en charge.
processing-instruction(nom) n'est pas pris en charge. Placez plutôt le nom entre guillemets.