SQL Server XML列の値をクエリする方法


127

RolesSQL ServerデータベースのXML列(と呼ばれます)に次のXMLが格納されています。

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

特定の役割を持つすべての行をリストします。この役割はパラメーターによって渡されます。

回答:


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

これらのページでは、T-SQLでXMLをクエリする方法について詳しく説明します。

t-sqlを使用したXMLフィールドのクエリ

SQL ServerでのXMLデータのフラット化

編集

もう少し試してみたところ、CROSS APPLYを使用したこの驚くべきクエリになりました。これは、すべての行(ロール)で、like式に入力した値を検索します...

このテーブル構造を考えると:

create table MyTable (Roles XML)

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

次のようにクエリできます。

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%'

ここでSQL Fiddleを確認できます:http ://sqlfiddle.com/#!18/dc4d2/1/ 0


5
それは私のすべての質問に[1]答えます、あなたの答えで何をしますか?
ビストロ

1
すばらしい答えです。私はこれに投票しますが、文字列はvarcharであると思います
AaA 2012

7
@ビストロについての[1]質問は本当に良い質問でした。これは、XMLから最初のロール値を選択することを意味し、これはAlphaサンプルxml での検索に対してのみ機能することを意味します。を検索しても、行は見つかりませんBeta
ミカエルエリクソン

1
私の場合、特定の属性値でノードを照会する必要がありました。この答えが私の解決策へと導きました。属性値を二重引用符で囲む必要がありました。
John N

XMLに名前空間がある場合、どのようにクエリを実行しますか?
FMFF、

36
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

クエリをwhere col like '%Beta%'使用できるように機能させる場合contains

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

フィールド名がRolesでテーブル名がtable1の場合、以下を使用して検索できます

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

これは良いですが、ここで使用して検索する方法はありlikeますか?forexample /root/role like ....
ビストロ

2
レニエルが説明したように.value('(/root/role)[1]', 'varchar(max)') like '%yourtext%'代わりに使用exists
AaA 2012

4
これを試しましたか?何を入力したかに関係なく、すべてを検索します@Role
ミカエルエリクソン

6

以下の簡単な作業を思いついたのも覚えやすいです:-)

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

1
それはあまりにも遅い検索につながるよう私たちは、文字列操作を経由して検索するようになっていません
マルコム・サルバドルを

5

あなたは次のことができます

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

明らかにこれはちょっとしたハックであり、正式なソリューションにはお勧めしません。ただし、SQL Server 2012のSQL Server Management StudioでXML列に対してアドホッククエリを実行する場合、この手法は非常に便利です。


2

便利なヒント。SQL Server XML列の値をクエリする(名前空間のあるXML)

例えば

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

例:パラメータのXML:

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

例:クエリ:

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

1

以下のステートメントを使用して、SQLテーブルのXMLの値を取得しました。

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'

XMLに名前空間定義が含まれていない場合はどうなりますか?
Muflix 2016

0

タグ全体または特定の値のみをクエリできます。ここでは、xml名前空間にワイルドカードを使用しています。

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
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.