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 를 사용하는이 놀라운 쿼리로 끝났습니다 . 이것은 당신이 같은 식에 넣은 값을 모든 행 (역할)에서 검색합니다 ...

이 테이블 구조가 주어지면 :

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]대답합니다. 당신의 대답은 무엇입니까?
Bistro

1
큰 대답, 나는 이것에 투표하지만 문자열은 내가 생각하는 varchar이어야합니다
AaA

7
@Bistro 질문은 [1]정말 좋은 질문이었습니다. 이는 XML에서 첫 번째 역할 값을 선택한다는 것을 의미하며 이는 Alpha샘플 xml에서 찾을 때만 작동합니다 . 검색하면 행을 찾을 수 없습니다 Beta.
Mikael Eriksson

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

필드 이름이 역할이고 테이블 이름이 table1 인 경우 다음을 사용하여 검색 할 수 있습니다.

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

이것은 좋습니다, 여기를 사용하여 검색 할 수있는 방법이 like있습니까? forexample /root/role like ....
Bistro

2
Leniel이 설명 하는 .value('(/root/role)[1]', 'varchar(max)') like '%yourtext%'대신 사용exists
AaA

4
이것을 시도 했습니까? 입력 한 내용에 관계없이 모든 것을 찾습니다 @Role.
Mikael Eriksson

6

나는 기억하기 쉬운 아래의 간단한 작업을 생각해 냈습니다 :-)

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

1
우리는 문자열 조작을 통해 검색하지 않아야합니다. 너무 느린 검색을 초래할 수 있습니다.
Malcolm Salvador

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

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.