在 SQL Server 中将多行动态组合成多列
Posted
技术标签:
【中文标题】在 SQL Server 中将多行动态组合成多列【英文标题】:Combine multiple rows into multiple columns dynamically in SQL Server 【发布时间】:2014-03-08 17:00:50 【问题描述】:我有一个大型数据库表,我需要使用 Microsoft SQL Server 在其上动态执行以下操作。
从这样的结果:
badge | name | Job | KDA | Match
- - - - - - - - - - - - - - - -
T996 | Darrien | AP | 3.0 | 20
T996 | Darrien | ADC | 2.8 | 16
T996 | Darrien | TOP | 5.0 | 120
使用 SQL 得到这样的结果:
badge | name | AP_KDA | AP_Match | ADC_KDA | ADC_Match | TOP_KDA | TOP_Match
- - - - - - - - -
T996 | Darrien | 3.0 | 20 | 2.8 | 16 | 5.0 | 120
即使有 30 行,它也会合并为 60 列的单行。
我目前可以通过硬编码(参见下面的示例)来做到这一点,但不是动态的。
Select badge,name,
(
SELECT max(KDA)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'AP')
) AP_KDA,
(
SELECT max(Match)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'AP')
) AP_Match,
(
SELECT max(KDA)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'ADC')
) ADC_KDA,
(
SELECT max(Match)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'ADC')
) ADC_Match,
(
SELECT max(KDA)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'TOP')
) TOP_KDA,
(
SELECT max(Match)
FROM table
WHERE (h.badge = badge) AND (h.name = name)
AND (Job = 'TOP')
) TOP_Match
from table h
我需要一个 MSSQL 语句,它允许我将多行合并为一行。第 3 列 (Job
) 内容将与第 4 列和第 5 列标题(KDA
和 Match
)合并成为一个新列。
所以,如果Job
有 6 个不同的值(比如 Job1
到 Job6
),那么结果将有 12 列,例如:Job1_KDA
、Job1_Match
、Job2_KDA
、@987654334 @ 等,按徽章和名称分组。
我需要一个可以循环遍历第 3 列数据的语句,因此我不需要硬编码(对每个可能的 Job
值重复查询)或使用临时表。
【问题讨论】:
大家好,我要问的问题是一个 mssql 语句,它允许我将多行合并为一行。第 3 列内容将与第 4,5 列标题合并,成为一个新列。因此,如果第 3 列中有 6 个不同的数据,那么它将产生 12 列。例如:Job1_KDA、Job1_Match、Job2_KDA、Job2_Match、Job3_KDA、Job3_Match、Job4_KDA、Job4_Match 这里有很多相关的问题。搜索[sql-server] pivot
以获取多个完全按照您的要求进行操作的示例。例如,this question 执行类似的类型操作(尽管它将六行转换为三行)。
【参考方案1】:
我会使用动态 sql,但这是 (http://sqlfiddle.com/#!6/a63a6/1/0) PIVOT 解决方案:
SELECT badge, name, [AP_KDa], [AP_Match], [ADC_KDA],[ADC_Match],[TOP_KDA],[TOP_Match] FROM
(
SELECT badge, name, col, val FROM(
SELECT *, Job+'_KDA' as Col, KDA as Val FROM @T
UNION
SELECT *, Job+'_Match' as Col,Match as Val FROM @T
) t
) tt
PIVOT ( max(val) for Col in ([AP_KDa], [AP_Match], [ADC_KDA],[ADC_Match],[TOP_KDA],[TOP_Match]) ) AS pvt
奖励:这是如何将 PIVOT 与动态 SQL (http://sqlfiddle.com/#!6/a63a6/7/0) 结合起来的,我更希望它更简单,没有 PIVOT,但这对我来说只是很好的锻炼:
SELECT badge, name, cast(Job+'_KDA' as nvarchar(128)) as Col, KDA as Val INTO #Temp1 FROM Temp
INSERT INTO #Temp1 SELECT badge, name, Job+'_Match' as Col, Match as Val FROM Temp
DECLARE @columns nvarchar(max)
SELECT @columns = COALESCE(@columns + ', ', '') + Col FROM #Temp1 GROUP BY Col
DECLARE @sql nvarchar(max) = 'SELECT badge, name, '+@columns+' FROM #Temp1 PIVOT ( max(val) for Col in ('+@columns+') ) AS pvt'
exec (@sql)
DROP TABLE #Temp1
【讨论】:
您好 Roman,感谢您提出解决方案。但这并没有解决我的情况,因为我需要列名来动态创建自己而不是对其进行硬编码。如何使 PIVOT 语句不是像 AP_KDA、AP_Match 这样的静态变量,而是像 Job+'_KDA'、Job+'_Match' 这样的变量? 我以为我给你的答案是“使用动态 sql”。您需要帮助来编写动态查询吗? PIVOT 是我写的,只是因为我很少使用它们,这对我来说很有趣。 看起来不错,但在最后一行出现错误。 DECLARE @sql nvarchar(max) = 'SELECT badge, name, '+@columns+' FROM #Temp1 PIVOT ( max(val) for Col in ('+@columns+') ) AS pvt' 它给了我这个错误。消息 139,级别 15,状态 1,行 0 无法将默认值分配给局部变量。 Msg 137, Level 15, State 2, Line 6 必须声明标量变量“@sql”。我搜索了错误解决方案,所有用户也提到需要将列转换为 nvarchar,所以我这样做了。我已经将所有列转换为 nvarchar,但仍然无法解决此错误。 nvm,我自己解决了,似乎是我里面的数据得到了 % 导致它出错了。现在它就像一个魅力,谢谢!【参考方案2】:将多行多列组合成一行并按 ID 分组
IF OBJECT_ID('usr_CUSTOMER') IS NOT NULL
DROP TABLE usr_CUSTOMER
--------------------------CRATE TABLE---------------------------------------------------
GO
SET ANSI_NULLS ON
GO
SET QUOTED_IDENTIFIER ON
GO
CREATE TABLE [dbo].[usr_CUSTOMER](
[Last_Name] [nvarchar](50) NULL,
[First_Name] [nvarchar](50) NULL,
[Middle_Name] [nvarchar](50) NOT NULL,
[ID] [int] NULL
) ON [PRIMARY]
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'gal', N'ornon', N'gili', 111)
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'porat', N'Yahel', N'LILl', 44444)
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'Shabtai', N'Or', N'Orya', 2222)
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'alex', N'levi', N'dolev', 33)
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'oren', N'cohen', N'ornini', 44444)
GO
INSERT [dbo].[usr_CUSTOMER] ([Last_Name], [First_Name], [Middle_Name], [ID]) VALUES (N'ron', N'ziyon', N'amir', 2222)
GO
----------------------------script---------------------------------------------
IF OBJECT_ID('tempdb..#TempString') IS NOT NULL
DROP TABLE #TempString
IF OBJECT_ID('tempdb..#tempcount') IS NOT NULL
DROP TABLE #tempcount
IF OBJECT_ID('tempdb..#tempcmbnition') IS NOT NULL
DROP TABLE #tempcmbnition
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
SELECT ID,
[Last_Name] + '#' + [First_Name] + '#' + ISNULL([Middle_Name], '') as StringRow
INTO #TempString
FROM [dbo].[usr_CUSTOMER]
ORDER BY StringRow
select distinct id
into #tempcount
from usr_CUSTOMER
CREATE TABLE [dbo].[#tempcmbnition](
[ID] [int] NULL,
[combinedString] [nvarchar](max) NULL
)
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
DECLARE @tableID table(ID int)
insert into @tableID(ID) (select distinct Id from #tempcount)
DECLARE @CNT int
SET @CNT = (select count(*) from @tableID)
declare @lastRow int
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
WHILE (@CNT >=1 )
BEGIN
SET @lastRow = (SELECT TOP 1 id FROM #tempcount ORDER BY id DESC)
DECLARE @combinedString VARCHAR(MAX)
set @combinedString = ''
SELECT @combinedString = COALESCE(@combinedString + '^ ', '') + StringRow
from #TempString
where ID = @lastRow
insert into #tempcmbnition (ID, [combinedString]) values(@lastRow ,@combinedString)
SET @CNT = @CNT-1
DELETE #tempcount where ID = @lastRow
END
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
-- if you what remove first char
-- UPDATE #tempcmbnition
-- SET combinedString = RIGHT(combinedString, LEN(combinedString) - 1)
--------------------------------------------------------------------------------
--------------------------------------------------------------------------------
select *from #TempString
select * from #tempcmbnition
【讨论】:
以上是关于在 SQL Server 中将多行动态组合成多列的主要内容,如果未能解决你的问题,请参考以下文章
当没有动态列时,将多列和多行的列连接成一个 varchar 值