SQL Server 2012:解析XML然后选择一个子节点以及连接所有子节点

时间:2016-11-28 20:25:20

标签: sql-server xml tsql xml-parsing cross-apply

我正在尝试解析XML列并尝试做两件事

  • 遍历节点的多个子节点,并根据1/0标志值选择一个节点。
  • 组合所有子节点以创建一个组合字段(分隔)

可运行代码块 - SQL Server 2012:

create table #temp (XMLData xml)

insert into #temp (XMLData)
values ('
<Report_Data>
  <Report_Entry>
    <IsActive>1</IsActive>
    <PID>111</PID>
    <Languages>
      <Language>German</Language>
      <speak>Y</speak>
      <read>Y</read>
      <write>Y</write>
    </Languages>
    <Languages>
      <Language>Spanish</Language>
      <speak>Y</speak>
      <read>N</read>
      <write>N</write>
    </Languages>
    <phone>
      <PhoneNumber>(101)111-1111</PhoneNumber>
      <PhoneType>Work</PhoneType>
      <IsPrimary>1</IsPrimary>
    </phone>
    <phone>
      <PhoneNumber>(101)111-2222</PhoneNumber>
      <PhoneType>Mobile</PhoneType>
      <IsPrimary>0</IsPrimary>
    </phone>
  </Report_Entry>
  <Report_Entry>
    <IsActive>1</IsActive>
    <PID>222</PID>
    <phone>
      <PhoneNumber>(101)222-1111</PhoneNumber>
      <PhoneType>Mobile</PhoneType>
      <IsPrimary>0</IsPrimary>
    </phone>
  </Report_Entry>
  <Report_Entry>
    <IsActive>1</IsActive>
    <PID>333</PID>
    <phone>
      <PhoneNumber>(101)333-1111</PhoneNumber>
      <PhoneType>Phone</PhoneType>
      <IsPrimary>0</IsPrimary>
    </phone>
    <phone>
      <PhoneNumber>(101)333-2222</PhoneNumber>
      <PhoneType>Mobile</PhoneType>
      <IsPrimary>1</IsPrimary>
    </phone>
    <location>
      <location-state>NY</location-state>
    </location>
    <location>
      <location-state>DC</location-state>
    </location>
  </Report_Entry>
</Report_Data>
')

select 
      c.value('IsActive[1]','varchar(1)') as IsActive
    , c.value('PID[1]','varchar(5)') as PID
    , case when c.value('phone[1]/IsPrimary[1]','int') = 1 then c.value('phone[1]/PhoneNumber[1]','varchar(15)') end as  PublicWorkPhone /** this condition needs to look at all sub nodes. this stops at the first one.  **/
    , c.value('location[1]','varchar(2)') as location
from   
    #temp 
    cross apply #temp.XMLData.nodes('/Report_Data/Report_Entry') as y(c)

drop table #temp
GO

我明白了:

IsActive PID   PublicWorkPhone location
-------- ----- --------------- --------
1        111   (101)111-1111   NULL
1        222   NULL            NULL
1        333   NULL            NY

但是,我需要这个:

IsActive PID   PublicWorkPhone location
-------- ----- --------------- --------
1        111   (101)111-1111   NULL
1        222   NULL            NULL
1        333   (101)333-2222   NY,DC

对于PID = 333,主要电话是(101)333-2222而不是空。 此外,位置应该是&#34; NY,DC&#34;不只是纽约。

我非常感谢您提供的任何帮助,以达到最佳效果。 谢谢

2 个答案:

答案 0 :(得分:1)

我的建议使用XQuery中的.nodes()来查找电话节点,即主要,以及XQuery - 函数data()你的位置。此函数将返回所有包含的文本部分,由空格分隔(顺便说一句:奇怪的是,一个人无法传递到分隔符char ...)。在您的情况下,我希望 solid-two-char 位置代码。只需用逗号替换空白即可...

SELECT re.value(N'IsActive[1]','bit') AS IsActive
      ,re.value(N'PID[1]','int') AS PID
      ,ph.value(N'PhoneNumber[1]','nvarchar(max)') AS PublicWorkPhone
      ,REPLACE(re.query(N'data(location/location-state)').value('.','nvarchar(max)'),' ',',') AS location
FROM #temp AS tmp
CROSS APPLY tmp.XMLData.nodes(N'/Report_Data/Report_Entry') AS A(re)
OUTER APPLY re.nodes(N'phone[IsPrimary=1]') AS B(ph);

答案 1 :(得分:0)

这应该可以解决问题:

WITH prep AS 
(
  SELECT 
    c.value('(IsActive[1]/text())[1]','char(1)') as IsActive,
    c.value('(PID[1]/text())[1]','varchar(5)') as PID,
    c.value('(phone[IsPrimary=1]/PhoneNumber/text())[1]', 'varchar(15)') AS PublicWorkPhone,
    location = cc.value('(text())[1]', 'varchar(1000)')
  from #temp 
  CROSS APPLY #temp.XMLData.nodes('/Report_Data/Report_Entry') as y(c)
  OUTER APPLY c.nodes('location/location-state') AS z(cc)
)
SELECT
  IsActive, 
  PID,
  PublicWorkPhone,
  Location = 
  STUFF ((
    SELECT ',' + location
    FROM prep pp
    WHERE p.PID = pp.PID
    FOR XML PATH('')),1,1,'')
FROM prep p
GROUP BY IsActive, PID, PublicWorkPhone;