使用 SQL 将 XML 结构转置/展平到列

2022-01-22 00:00:00 pivot xml tsql sql-server xquery-sql

我正在使用 SQL Server (2008/2012),并且我知道从大量搜索中得到了类似的答案,但是我似乎无法找到适合我的案例的示例/指针.

I am using SQL Server (2008/2012) and I know there are similar answers from lots of searching, however I can't seem to find the appropriate example/pointers for my case.

我在 SQL Server 表中有一个 XML 列,其中包含这些数据:

I have an XML column in a SQL Server table holding this data:

<Items>
 <Item>
  <FormItem>
    <Text>FirstName</Text>
    <Value>My First Name</Value>
  </FormItem>
  <FormItem>
    <Text>LastName</Text>
    <Value>My Last Name</Value>
  </FormItem>
  <FormItem>
    <Text>Age</Text>
    <Value>39</Value>
  </FormItem>
 </Item>
 <Item>
  <FormItem>
    <Text>FirstName</Text>
    <Value>My First Name 2</Value>
  </FormItem>
  <FormItem>
    <Text>LastName</Text>
    <Value>My Last Name 2</Value>
  </FormItem>
  <FormItem>
    <Text>Age</Text>
    <Value>40</Value>
  </FormItem>
 </Item>
</Items>

所以即使 <FormItem> 的结构是相同的,我也可以拥有多组(通常不超过 20-30 个)表单项..

So even though the structure of <FormItem> is going to be the same, I can have multiple (most commonly no more than 20-30) sets of form items..

我实际上是在尝试以以下格式从 SQL 返回查询,即基于/FormItem/Text 的动态列:

I am essentially trying to return a query from SQL in the format below, i.e. dynamic columns based on /FormItem/Text:

FirstName         LastName         Age    ---> More columns as new `<FormItem>` are returned
My First Name     My Last Name     39          Whatever value etc..
My First Name 2   My Last Name 2   40          

所以,此刻我有以下几点:

So, at the moment I had the following:

select 
    Tab.Col.value('Text[1]','nvarchar(100)') as Question,
    Tab.Col.value('Value[1]','nvarchar(100)') as Answer
from
    @Questions.nodes('/Items/Item/FormItem') Tab(Col)

当然,这并没有将我的 XML 行转换为列,而且显然是用字段固定的.我一直在尝试各种动态 SQL"方法,其中 SQL 执行不同的选择(在我的情况下)<Text> 节点,然后使用某种 Pivot?但我似乎找不到神奇的组合来返回我需要的结果作为每一行的动态列集(<Items><集合中的 /code>).

Of course that hasn't transposed my XML rows into columns, and obviously is fixed with fields anyway.. I have been trying various "Dynamic SQL" approaches where the SQL performs a distinct selection of (in my case) the <Text> node, and then uses some sort of Pivot? but I couldn't seem to find the magic combination to return the results I need as a dynamic set of columns for each row (<Item> within the collection of <Items>).

我相信看到这么多非常相似的例子是可以做到的,但是我又找不到解决方案!

I'm sure it can be done having seen so many very similar examples, however again the solution eludes me!

感谢您的帮助!

推荐答案

解析 XML 相当昂贵,因此您可以创建一个具有名称-值的临时表,而不是解析一次来构建动态查询和一次获取数据列表,然后将其用作动态数据透视查询的来源.
dense_rank 用于创建 ID 以进行旋转.
为了在动态查询中构建列列表,它使用 for xml path('') 技巧.

Parsing the XML is fairly expensive so instead of parsing once to build a dynamic query and once to get the data you can create a temporary table with a Name-Value list and then use that as the source for a dynamic pivot query.
dense_rank is there to create the ID to pivot around.
To build the column list in the dynamic query it uses the for xml path('') trick.

此解决方案要求您的表具有主键 (ID).如果您在变量中包含 XML,则可以稍微简化一下.

This solution requires that your table has a primary key (ID). If you have the XML in a variable it can be somewhat simplified.

select dense_rank() over(order by ID, I.N) as ID,
       F.N.value('(Text/text())[1]', 'varchar(max)') as Name,
       F.N.value('(Value/text())[1]', 'varchar(max)') as Value
into #T
from YourTable as T
  cross apply T.XMLCol.nodes('/Items/Item') as I(N)
  cross apply I.N.nodes('FormItem') as F(N)

declare @SQL nvarchar(max)
declare @Col nvarchar(max)

select @Col = 
  (
  select distinct ','+quotename(Name)
  from #T
  for xml path(''), type
  ).value('substring(text()[1], 2)', 'nvarchar(max)')

set @SQL = 'select '+@Col+'
            from #T
            pivot (max(Value) for Name in ('+@Col+')) as P'

exec (@SQL)

drop table #T

SQL 小提琴

相关文章