Metodo nodes() (tipo di dati xml)
Si applica a: SQL Server Database SQL di Azure Istanza gestita di SQL di Azure
È possibile usare il metodo nodes() per suddividere un'istanza del tipo di dati xml in un insieme di dati relazionali. Tale metodo consente di identificare i nodi sui quali verrà eseguito il mapping a una nuova riga.
Per ogni istanza del tipo di dati xml esiste un nodo di contesto fornito implicitamente. Nel caso di un'istanza XML archiviata in una colonna o variabile, questo nodo è il nodo del documento, che costituisce il nodo implicito di livello più alto per ogni istanza del tipo di dati xml.
Il metodo nodes() restituisce un set di righe che contiene copie logiche delle istanze XML originali. In tali copie logiche il nodo di contesto di ogni istanza di riga viene impostato su uno dei nodi identificati tramite l'espressione della query, in modo che le query successive possano tener conto di questi nodi di contesto.
Dal set di righe è possibile recuperare più valori. È ad esempio possibile applicare il metodo value() al set di righe restituito dal metodo nodes() e recuperare più valori dall'istanza XML originale. Quando viene applicato a un'istanza XML, il metodo value() restituisce un solo valore.
Sintassi
nodes (XQuery) as Table(Column)
Argomenti
XQuery
Valore letterale stringa costituito da un'espressione XQuery. Se l'espressione della query costruisce nodi, questi ultimi verranno esposti nel set di righe risultante. Se l'espressione di query restituisce una sequenza vuota, il set di righe sarà vuoto. Se l'espressione della query restituisce staticamente una sequenza che contiene valori atomici, anziché nodi, verrà generato un errore statico.
Table(Column)
Nome della tabella e della colonna per il set di righe risultante.
Osservazioni:
Si consideri ad esempio la tabella seguente:
T (ProductModelID INT, Instructions XML)
In tale tabella è archiviato il seguente documento di istruzioni per la produzione, di cui è illustrato solo un frammento. Si noti che nel documento sono indicati tre centri di produzione.
<root>
<Location LocationID="10"...>
<step>...</step>
<step>...</step>
...
</Location>
<Location LocationID="20" ...>
...
</Location>
<Location LocationID="30" ...>
...
</Location>
</root>
Una chiamata al metodo nodes()
con l'espressione di query /root/Location
restituirebbe un set di righe con tre righe, ognuna contenente una copia logica del documento XML originale e con l'elemento di contesto impostato su uno dei nodi <Location>
:
Product
ModelID Instructions
----------------------------------
1 <root><Location LocationID="10" ... />
<Location LocationID="20" ... />
<Location LocationID="30" .../></root>
1 <root><Location LocationID="10" ... />
<Location LocationID="20" ... />
<Location LocationID="30" .../></root>
1 <root><Location LocationID="10" ... />
<Location LocationID="20" ... />
<Location LocationID="30" .../></root>
Per eseguire query su tale set di righe, è possibile usare i metodi con tipo di dati xml. La query seguente estrae il sottoalbero dell'elemento di contesto per ogni riga generata:
SELECT T2.Loc.query('.')
FROM T
CROSS APPLY Instructions.nodes('/root/Location') AS T2(Loc)
Questa è la formula:
ProductModelID Instructions
----------------------------------
1 <Location LocationID="10" ... />
1 <Location LocationID="20" ... />
1 <Location LocationID="30" .../>
Nel set di righe restituito sono state mantenute le informazioni sul tipo. È possibile applicare metodi con tipo di dati xml, ad esempio query(), value(), exist() e nodes() al risultato di un metodo nodes(). Non è tuttavia possibile applicare il metodo modify() per modificare l'istanza XML.
Inoltre, il nodo di contesto nel set di righe non può essere materializzato, ovvero non può essere usato in un'istruzione SELECT. Può essere tuttavia utilizzato nelle istruzioni IS NULL e COUNT(*).
Gli scenari d'uso del metodo nodes() sono gli stessi dell'uso di OPENXML (Transact-SQL), che fornisce una vista del set di righe del documento XML. Non è tuttavia necessario usare cursori quando si usa il metodo nodes() su una tabella che contiene più righe di documenti XML.
Il set di righe restituito dal metodo nodes() è senza nome, quindi deve essere denominato in modo esplicito tramite alias.
La funzione nodes() non può essere applicata direttamente ai risultati di una funzione definita dall'utente. Per usare la funzione nodes () con il risultato di una funzione scalare definita dall'utente è possibile:
- Assegnare il risultato della funzione definita dall'utente a una variabile
- Usare una tabella derivata per assegnare un alias di colonna al valore restituito dalla funzione definita dall'utente e quindi usare
CROSS APPLY
per eseguire la selezione dall'alias.
Nell'esempio seguente viene illustrato un modo per utilizzare CROSS APPLY
per selezionare il risultato di una funzione definita dall'utente.
USE AdventureWorks;
GO
CREATE FUNCTION XTest()
RETURNS XML
AS
BEGIN
RETURN '<document/>';
END;
GO
SELECT A2.B.query('.')
FROM
(SELECT dbo.XTest()) AS A1(X)
CROSS APPLY X.nodes('.') A2(B);
GO
DROP FUNCTION XTest;
GO
Esempi
Utilizzo del metodo nodes() su una variabile di tipo xml
Nell'esempio seguente viene usato un documento XML con un elemento di livello principale <Root
> e tre elementi figlio <row
>. La query utilizza il metodo nodes()
per impostare nodi di contesto separati, uno per ogni elemento <row
>. Il metodo nodes()
restituisce un set di righe con tre righe. Ogni riga contiene una copia logica dell'istanza XML originale e ogni nodo di contesto identifica un diverso elemento <row
> nel documento originale.
La query restituisce quindi il nodo di contesto di ogni riga:
DECLARE @x XML
SET @x='<Root>
<row id="1"><name>Larry</name><oflw>some text</oflw></row>
<row id="2"><name>moe</name></row>
<row id="3" />
</Root>'
SELECT T.c.query('.') AS result
FROM @x.nodes('/Root/row') T(c)
GO
Nel risultato dell'esempio seguente, il metodo della query restituisce l'elemento di contesto e il relativo contenuto:
<row id="1"><name>Larry</name><oflw>some text</oflw></row>
<row id="2"><name>moe</name></row>
<row id="3"/>
Se si applica la funzione di accesso padre ai nodi di contesto, viene restituito l'elemento <Root
> per tutti e tre:
SELECT T.c.query('..') AS result
FROM @x.nodes('/Root/row') T(c)
GO
Questa è la formula:
<Root>
<row id="1"><name>Larry</name><oflw>some text</oflw></row>
<row id="2"><name>moe</name></row>
<row id="3" />
</Root>
<Root>
<row id="1"><name>Larry</name><oflw>some text</oflw></row>
<row id="2"><name>moe</name></row>
<row id="3" />
</Root>
<Root>
<row id="1"><name>Larry</name><oflw>some text</oflw></row>
<row id="2"><name>moe</name></row>
<row id="3" />
</Root>
Utilizzo del metodo nodes() in una colonna di tipo xml
In questo esempio vengono usati documenti di istruzioni per la produzione di biciclette, archiviati nella colonna di tipo xml Instructions della tabella ProductModel.
Nell'esempio seguente il metodo nodes()
viene eseguito sulla colonna Instructions
di tipo xml della tabella ProductModel
.
Il metodo nodes()
imposta gli elementi <Location
> come nodi di contesto specificando il percorso /MI:root/MI:Location
. Il set di righe risultante include copie logiche del documento originale, una per ogni nodo <Location
> del documento, con il nodo di contesto impostato sull'elemento <Location
>. La funzione nodes()
restituisce quindi un set di nodi di contesto <Location
>.
Il metodo query()
su questo set di righe richiede self::node
e restituisce l'elemento <Location>
di ogni riga.
In questo esempio la query imposta ogni elemento <Location
> come nodo di contesto nel documento di istruzioni per la produzione di uno specifico modello di prodotto. È possibile usare questi nodi di contesto per recuperare valori quali i seguenti:
Trovare gli ID località in ogni <
Location
>Recuperare i passaggi di produzione (elementi figlio <
step
>) in ogni <Location
>
La query seguente restituisce l'elemento di contesto. Nel metodo '.'
viene specificata la sintassi abbreviata per self::node()
, ovvero query()
.
Notare quanto segue:
Il metodo
nodes()
viene applicato alla colonna Instructions e restituisce il set di righeT (C)
, il quale contiene copie logiche del documento originale di istruzioni per la produzione con/root/Location
come elemento di contesto.CROSS APPLY applica
nodes()
a ogni riga nella colonnaProductModel
e restituisce solo le righe che producono un set di risultati.SELECT C.query('.') as result FROM Production.ProductModel CROSS APPLY Instructions.nodes(' declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions"; /MI:root/MI:Location') as T(C) WHERE ProductModelID=7
Di seguito è riportato il risultato parziale:
<MI:Location LocationID="10" ...> <MI:step ... /> ... </MI:Location> <MI:Location LocationID="20" ... > <MI:step ... /> ... </MI:Location> ...
Applicazione del metodo nodes() al set di righe restituito da un altro metodo nodes()
Il codice seguente esegue una query sui documenti XML con le istruzioni per la produzione archiviati nella colonna Instructions
della tabella ProductModel
. La query restituisce un set di righe che contiene l'ID del modello di prodotto, i centri di produzione e i passaggi di produzione.
Notare quanto segue:
Il metodo
nodes()
viene applicato alla colonnaInstructions
e restituisce il set di righeT1 (Locations)
, il quale contiene copie logiche del documento originale di istruzioni per la produzione con/root/Location
come elemento di contesto.Il metodo
nodes()
viene applicato al set di righeT1 (Locations)
e restituisce il set di righeT2 (steps)
, il quale contiene copie logiche del documento originale di istruzioni per la produzione con/root/Location/step
come elemento di contesto.
SELECT ProductModelID, Locations.value('./@LocationID','int') AS LocID,
steps.query('.') AS Step
FROM Production.ProductModel
CROSS APPLY Instructions.nodes('
declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
/MI:root/MI:Location') AS T1(Locations)
CROSS APPLY T1.Locations.nodes('
declare namespace MI="https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions";
./MI:step ') AS T2(steps)
WHERE ProductModelID=7
GO
Questa è la formula:
ProductModelID LocID Step
----------------------------
7 10 <step ... />
7 10 <step ... />
...
7 20 <step ... />
7 20 <step ... />
7 20 <step ... />
...
Nella query il prefisso MI
viene dichiarato due volte. In alternativa, è possibile utilizzare WITH XMLNAMESPACES
per dichiarare il prefisso una volta sola e utilizzarlo nella query:
WITH XMLNAMESPACES (
'https://schemas.microsoft.com/sqlserver/2004/07/adventure-works/ProductModelManuInstructions' AS MI)
SELECT ProductModelID, Locations.value('./@LocationID','int') AS LocID,
steps.query('.') AS Step
FROM Production.ProductModel
CROSS APPLY Instructions.nodes('
/MI:root/MI:Location') AS T1(Locations)
CROSS APPLY T1.Locations.nodes('
./MI:step ') as T2(steps)
WHERE ProductModelID=7
GO
Vedi anche
Aggiungere spazi dei nomi alle query con WITH XMLNAMESPACES
Creare istanze di dati XML
metodi con tipo di dati xml