sql - 具有多种数据类型的动态数据透视表

标签 sql pivot-table

我有一个关于数据透视表的技巧问题:

我有一张看起来像这样的表:

id  table   object  name     type   nvarchar    date    int     bit  
1   1       2       name     1      tables      NULL    NULL    NULL  
2   1       2       name     1      columns     NULL    NULL    NULL  
3   1       2       name     1      datatypes   NULL    NULL    NULL  
4   1       2       name     1      _users      NULL    NULL    NULL  
1   1       3       active   3      NULL        NULL    NULL    1  
2   1       3       active   3      NULL        NULL    NULL    1  
3   1       3       active   3      NULL        NULL    NULL    1  
4   1       3       active   3      NULL        NULL    NULL    1  

输出应该是这样的:

id   name       active   
1    tables     1  
2    columns    1  
3    datatypes  1  
4    _users     1  

根据“类型”,我应该将列中的正确数据放入其中,这些列的格式为 nvarchar、bit、datetime、int 等。

“id”是行 ID,“name, active”来自名称列,值来自 nvarchar、date、int 和 bit 列。

更新:像 nvarchar、date、int 和 bit(以及大多数其他 SQL 格式)这样的列实际上包含这种类型的数据。 “类型”列给出了哪一列包含要使用的数据,所以如果“类型”是“1”,那么我想使用“nvarchar”,如果“类型”是“3”,那么我想使用“位” "其中确实包含一点而不是 nvarchar。在 Pivot 中,我想在“事件”列下显示该位,如果我在示例中有第 3 列(名称),例如“activation_date”,我想看到第三列的值(type = 2)来自日期专栏。

我迷路了,请帮忙

最佳答案

假设每一行只有一个非空列:

with cte as (
    select
        id,
        name,
        coalesce(
            [nvarchar],
            convert(nvarchar(max), [date], 120),
            cast([int] as nvarchar(max)),
            cast([bit] as nvarchar(max))
        ) as value
    from Table1 as t
)
select
    id,
    max(case when [name] = 'name' then value end) as [name],
    max(case when [name] = 'active' then value end) as [active]
from cte
group by id

sql fiddle demo

但我必须警告你,这种类型的数据库模式并不是使用 SQL 的最佳方式。

如果您想在不对列进行硬编码的情况下动态执行此操作:

declare @stmt nvarchar(max)

select @stmt =
   isnull(@stmt + ', ', '') +
   'max(case when [name] = ''' + name + ''' then value end) as ' + quotename([name])
from (select distinct [name] from Table1) as t

select @stmt = '
with cte as (
    select
        id,
        name,
        coalesce(
            [nvarchar],
            convert(nvarchar(max), [date], 120),
            cast([int] as nvarchar(max)),
            cast([bit] as nvarchar(max))
        ) as value
    from Table1 as t
)
select
    id, ' + @stmt + '
from cte
group by id
'

exec sp_executesql
    @stmt = @stmt

sql fiddle demo

如果你有这样的映射表:

name       value
--------------------
name       nvarchar
active     bit

你可以使用这个查询:

declare @stmt nvarchar(max)

select @stmt =
   isnull(@stmt + ', ', '') +
   'max(case when [name] = ''' + name + ''' then [' + value + '] end) as ' + quotename([name])
from Mapping

select @stmt = '
select
    id, ' + @stmt + '
from Table1
group by id
'

exec sp_executesql
    @stmt = @stmt

sql fiddle demo

关于sql - 具有多种数据类型的动态数据透视表,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18849515/

相关文章:

mysql - 动态数据透视表列 mysql

基于列名前缀的 Pandas 数据透视列

Python Pandas 按多个类别和年份交叉表

sql - 在 golang 中使用准备好的语句时,是否总是需要延迟 stmt.close()?

mysql - 在多个连接表和相关表中查找特定结果

sql - 检索 CLOB 字段期间出现 DB2 查询错误

python - 使用 Openpyxl 创建数据透视表

mysql - INNER JOIN 后从多个表中获取最大日期

php - mysql连接失败

excel - 众多枢轴领域中的一个崩溃