Wie kann ich einen Wert in der SQL Server-XML-Spalte abfragen?


127

Ich habe folgendes XML in einer XML-Spalte (aufgerufen Roles) in einer SQL Server-Datenbank gespeichert.

<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>

Ich möchte alle Zeilen auflisten, die eine bestimmte Rolle in ihnen haben. Diese Rolle wird vom Parameter übergeben.

Antworten:


198
select
  Roles
from
  MyTable
where
  Roles.value('(/root/role)[1]', 'varchar(max)') like 'StringToSearchFor'

Auf diesen Seiten erfahren Sie mehr über das Abfragen von XML in T-SQL:

Abfragen von XML-Feldern mit t-sql

Reduzieren von XML-Daten in SQL Server

BEARBEITEN

Nachdem ich ein bisschen mehr damit gespielt hatte, kam ich zu dieser erstaunlichen Abfrage, die CROSS APPLY verwendet . Dieser durchsucht jede Zeile (Rolle) nach dem Wert, den Sie in Ihren ähnlichen Ausdruck einfügen ...

Angesichts dieser Tabellenstruktur:

create table MyTable (Roles XML)

insert into MyTable values
('<root>
   <role>Alpha</role>
   <role>Gamma</role>
   <role>Beta</role>
</root>')

Wir können es so abfragen:

select * from 

(select 
       pref.value('(text())[1]', 'varchar(32)') as RoleName
from 
       MyTable CROSS APPLY

       Roles.nodes('/root/role') AS Roles(pref)
)  as Result

where RoleName like '%ga%'

Sie können die SQL-Geige hier überprüfen: http://sqlfiddle.com/#!18/dc4d2/1/0


5
Es beantwortet alle meine Fragen, was macht [1]Ihre Antwort?
Bistro

1
Tolle Antwort, ich stimme für diese, aber die Saite sollte varchar sein, denke ich
AaA

7
@Bistro Fragen [1]war eine wirklich gute Frage. Dies bedeutet, dass Sie den ersten Rollenwert aus dem XML auswählen. Dies bedeutet, dass dies nur zum Auffinden Alphain Ihrer Beispiel-XML funktioniert . Die Zeile wird nicht gefunden, wenn Sie nach suchen Beta.
Mikael Eriksson

1
In meinem Fall musste ich Knoten mit einem bestimmten Attributwert abfragen. Diese Antwort war der Weg zu meiner Lösung. Ich musste nur den Attributwert in doppelte Anführungszeichen setzen.
John N

Wenn das XML einen Namespace hat, wie fragen wir ihn ab?
FMFF

35
declare @T table(Roles xml)

insert into @T values
('<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>')

declare @Role varchar(10)

set @Role = 'Beta'

select Roles
from @T
where Roles.exist('/root/role/text()[. = sql:variable("@Role")]') = 1

Wenn Sie möchten, dass die Abfrage so funktioniert, wie where col like '%Beta%'Sie sie verwenden könnencontains

declare @T table(Roles xml)

insert into @T values
('<root>
   <role>Alpha</role>
   <role>Beta</role>
   <role>Gamma</role>
</root>')

declare @Role varchar(10)

set @Role = 'et'

select Roles
from @T
where Roles.exist('/root/role/text()[contains(., sql:variable("@Role"))]') = 1

13

Wenn Ihr Feldname Rollen und der Tabellenname Tabelle1 ist, können Sie Folgendes zur Suche verwenden

DECLARE @Role varchar(50);
SELECT * FROM table1
WHERE Roles.exist ('/root/role = sql:variable("@Role")') = 1

Das ist gut, gibt es hier eine Möglichkeit, die Verwendung zu suchen like? forexample /root/role like ....
Bistro

2
Verwenden Sie .value('(/root/role)[1]', 'varchar(max)') like '%yourtext%'statt existswie Leniel erklärte
AaA

4
Hast du das versucht? Es findet alles, unabhängig davon, was Sie eingeben @Role.
Mikael Eriksson

6

Ich habe mir unten eine einfache Arbeit ausgedacht, an die man sich auch leicht erinnern kann :-)

select * from  
(select cast (xmlCol as varchar(max)) texty
 from myTable (NOLOCK) 
) a 
where texty like '%MySearchText%'

1
Wir sollten nicht über String-Manipulationen suchen, da dies zu viel zu langsamen Suchvorgängen führen würde
Malcolm Salvador

5

Sie könnten Folgendes tun

declare @role varchar(100) = 'Alpha'
select * from xmltable where convert(varchar(max),xmlfield) like '%<role>'+@role+'</role>%'

Offensichtlich ist dies ein kleiner Hack und ich würde es für keine formalen Lösungen empfehlen. Ich finde diese Technik jedoch sehr nützlich, wenn Ad-hoc-Abfragen für XML-Spalten in SQL Server Management Studio für SQL Server 2012 durchgeführt werden.


2

Nützlicher Tipp. Fragen Sie einen Wert in der SQL Server-XML-Spalte ab (XML mit Namespace).

z.B

Table [dbo].[Log_XML] contains columns Parametrs (xml),TimeEdit (datetime)

zB XML in Parametern:

<ns0:Record xmlns:ns0="http://Integration"> 
<MATERIAL>10</MATERIAL> 
<BATCH>A1</BATCH> 
</ns0:Record>

zB Abfrage:

select
 Parametrs,TimeEdit
from
 [dbo].[Log_XML]
where
 Parametrs.value('(//*:Record/BATCH)[1]', 'varchar(max)') like '%A1%'
 ORDER BY TimeEdit DESC

1

Ich habe die folgende Anweisung verwendet, um die Werte im XML in der SQL-Tabelle abzurufen

with xmlnamespaces(default 'http://test.com/2008/06/23/HL.OnlineContract.ValueObjects')
select * from (
select
            OnlineContractID,
            DistributorID,
            SponsorID,
    [RequestXML].value(N'/OnlineContractDS[1]/Properties[1]/Name[1]', 'nvarchar(30)') as [Name]
   ,[RequestXML].value(N'/OnlineContractDS[1]/Properties[1]/Value[1]', 'nvarchar(30)') as [Value]
     ,[RequestXML].value(N'/OnlineContractDS[1]/Locale[1]', 'nvarchar(30)') as [Locale]
from [OnlineContract]) as olc
where olc.Name like '%EMAIL%' and olc.Value like '%EMAIL%' and olc.Locale='UK EN'

Was ist, wenn XML keine Namespace-Definition enthält?
Muflix

0

Sie können das gesamte Tag oder nur den spezifischen Wert abfragen. Hier verwende ich einen Platzhalter für die XML-Namespaces.

declare @myDoc xml
set @myDoc = 
'<Root xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://stackoverflow.com">
    <Child>my value</Child>
 </Root>'

select @myDoc.query('/*:Root/*:Child') -- whole tag
select @myDoc.value('(/*:Root/*:Child)[1]', 'varchar(255)') -- only value
Durch die Nutzung unserer Website bestätigen Sie, dass Sie unsere Cookie-Richtlinie und Datenschutzrichtlinie gelesen und verstanden haben.
Licensed under cc by-sa 3.0 with attribution required.