如何在SQL Server XML列中查询值


127

我将以下XML存储在RolesSQL Server数据库的XML列(称为)中。

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

我想列出所有在其中具有特定角色的行。此角色由参数传递。

Answers:


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]您的回答是什么?
小酒馆2012年

1
好的答案,我赞成这一点,但是我猜字符串应该是varchar
AaA

7
@Bistro询问[1]是一个非常好的问题。这意味着您从XML中选择了第一个角色值,这意味着这仅适用于Alpha在示例xml中查找。如果搜索,它将找不到该行Beta
Mikael Eriksson'4

1
就我而言,我必须查询具有特定属性值的节点。这个答案是导致我解决方案的原因。我只需要在属性值两边加上双引号。
约翰N

如果XML具有名称空间,我们如何查询它?
FMFF '18年

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 ....
小酒馆2012年

2
使用.value('(/root/role)[1]', 'varchar(max)') like '%yourtext%'而不是existsLeniel解释的内容
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>%'

显然,这有点hack,我不建议在任何正式解决方案中使用它。但是,我发现在针对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 '16

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
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.