如何查询 SQL Server XML 列中的值

Posted

技术标签:

【中文标题】如何查询 SQL Server XML 列中的值【英文标题】:How can I query a value in SQL Server XML column 【发布时间】:2012-05-07 20:13:48 【问题描述】:

我在 SQL Server 数据库的 XML 列(称为Roles)中存储了以下 XML。

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

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

【问题讨论】:

【参考方案1】:
select
  Roles
from
  MyTable
where
  Roles.value('(/root/role)[1]', 'varchar(max)') like 'StringToSearchFor'

如果您的列不是XML,则需要对其进行转换。您还可以使用其他语法来查询 XML 数据的某些属性。这是一个例子......

假设数据列有这个:

<Utilities.CodeSystems.CodeSystemCodes iid="107" CodeSystem="2" Code="0001F" CodeTags="-19-"..../>

...而您只想要CodeSystem = 2 的那些,那么您的查询将是:

select 
  [data] 
from
  [dbo].[CodeSystemCodes_data]
  
where
  CAST([data] as XML).value('(/Utilities.CodeSystems.CodeSystemCodes/@CodeSystem)[1]', 'varchar(max)') = '2'

这些页面将向您展示有关如何在 T-SQL 中查询 XML 的更多信息:

Querying XML fields using t-sql

Flattening XML Data in SQL Server

编辑

在玩了一会儿之后,我最终得到了这个使用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

【讨论】:

它回答了我所有的问题,[1] 在你的回答中做了什么? 很好的答案,我投票给这个,但我猜字符串应该是 varchar @Bistro 询问[1] 是一个非常好的问题。这意味着您从 XML 中选择第一个角色值,这意味着这只适用于在您的示例 xml 中查找Alpha。如果您搜索Beta,它将找不到该行。 就我而言,我必须查询具有特定属性值的节点。这个答案是我解决方案的线索。我只需要在属性值周围加上双引号。 如果XML有命名空间,我们如何查询呢?【参考方案2】:
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

【讨论】:

【参考方案3】:

如果您的字段名称是 Roles 并且表名称是 table1 您可以使用以下搜索

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

【讨论】:

这很好,这里有什么方法可以搜索使用like吗? forexample /root/role like .... 使用 .value('(/root/role)[1]', 'varchar(max)') like '%yourtext%' 而不是 exists,正如 Leniel 解释的那样 你试过了吗?无论您在 @Role 中输入什么内容,它都能找到所有内容。【参考方案4】:

我想出了一个简单的解决方法,它也很容易记住:-)

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

【讨论】:

我们不应该通过字符串操作进行搜索,因为这会导致搜索速度太慢【参考方案5】:

您可以执行以下操作

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

显然,这有点小题大做,我不建议将它用于任何正式的解决方案。但是,我发现在 SQL Server Management Studio for SQL Server 2012 中对 XML 列进行即席查询时,这种技术非常有用。

【讨论】:

【参考方案6】:

有用的提示。查询 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

【讨论】:

【参考方案7】:

我使用下面的语句来检索 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 不包含命名空间定义怎么办?【参考方案8】:

您可以查询整个标签,也可以只查询特定的值。这里我为 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://***.com">
    <Child>my value</Child>
 </Root>'

select @myDoc.query('/*:Root/*:Child') -- whole tag
select @myDoc.value('(/*:Root/*:Child)[1]', 'varchar(255)') -- only value

【讨论】:

【参考方案9】:

如果你想找到“Alpha”之外的其他节点,查询应该是这样的:

select Roles from MyTable where Roles.exist('(/*:root/*:role[contains(.,"Beta")])') = 1

【讨论】:

以上是关于如何查询 SQL Server XML 列中的值的主要内容,如果未能解决你的问题,请参考以下文章

如何从SQL中的列值中提取特定部分(Redshift平台)

在 SQL Server XML 列中,如何验证相同的值是不是存在于多行中

如何在使用 SQL 的重复搜索中排除其他值中的值

动态获取 SQL Server 中不同列中的 XML 数据类型值

如何将另一行中的值插入列中 - SQL Server

SQL 查询:如何在排名列值中使用时间戳列