行转列,列转行是我们在开发过程中经常碰到的问题。行转列一般通过CASE WHEN 语句来实现,也可以通过 SQL SERVER 的运算符PIVOT来实现。用传统的方法,比较好理解。层次清晰,而且比较习惯。但是PIVOT 、UNPIVOT提供的语法比一系列复杂的SELECT…CASE 语句中所指定的语法更简单、更具可读性。下面我们通过几个简单的例子来介绍一下列转行、行转列问题。
SELECT CONVERT(VARCHAR(10), CreateTime, 120) AS CreateTime, CASE PayType WHEN'支付宝'THENSUM(Money) ELSE0ENDAS'支付宝', CASE PayType WHEN'手机短信'THENSUM(Money) ELSE0ENDAS'手机短信', CASE PayType WHEN'工商银行卡'THENSUM(Money) ELSE0ENDAS'工商银行卡', CASE PayType WHEN'建设银行卡'THENSUM(Money) ELSE0ENDAS'建设银行卡' FROM Inpours GROUPBY CreateTime, PayType
如图所示,我们这样只是得到了这样的输出结果,还需进一步处理,才能得到想要的结果
SELECT CreateTime, ISNULL(SUM([支付宝]) , 0) AS [支付宝], ISNULL(SUM([手机短信]) , 0) AS [手机短信], ISNULL(SUM([工商银行卡]), 0) AS [工商银行卡], ISNULL(SUM([建设银行卡]), 0) AS [建设银行卡] FROM ( SELECT CONVERT(VARCHAR(10), CreateTime, 120) AS CreateTime, CASE PayType WHEN'支付宝'THENSUM(Money) ELSE0ENDAS'支付宝' , CASE PayType WHEN'手机短信'THENSUM(Money) ELSE0ENDAS'手机短信', CASE PayType WHEN'工商银行卡'THENSUM(Money) ELSE0ENDAS'工商银行卡', CASE PayType WHEN'建设银行卡'THENSUM(Money) ELSE0ENDAS'建设银行卡' FROM Inpours GROUPBY CreateTime, PayType ) T GROUPBY CreateTime
其实行转列,关键是要理清逻辑,而且对分组(Group by)概念比较清晰。上面两个列子基本上就是行转列的类型了。但是有个问题来了,上面是我为了说明弄的一个简单列子。实际中,可能支付方式特别多,而且逻辑也复杂很多,可能涉及汇率、手续费等等(曾经做个这样一个),如果支付方式特别多,我们的CASE WHEN 会弄出一大堆,确实比较恼火,而且新增一种支付方式,我们还得修改脚本如果把上面的脚本用动态SQL改写一下,我们就能轻松解决这个问题
DECLARE @cmdText VARCHAR(8000); DECLARE @tmpSql VARCHAR(8000); SET @cmdText = 'SELECT CONVERT(VARCHAR(10), CreateTime, 120) AS CreateTime,' + CHAR(10); SELECT @cmdText = @cmdText + ' CASE PayType WHEN ''' + PayType + ''' THEN SUM(Money) ELSE 0 END AS ''' + PayType + ''',' + CHAR(10) FROM (SELECTDISTINCT PayType FROM Inpours ) T
SET @cmdText = LEFT(@cmdText, LEN(@cmdText) -2) --注意这里,如果没有加CHAR(10) 则用LEFT(@cmdText, LEN(@cmdText) -1)
SET @cmdText = @cmdText + ' FROM Inpours GROUP BY CreateTime, PayType ';
SET @tmpSql ='SELECT CreateTime,' + CHAR(10); SELECT @tmpSql = @tmpSql + ' ISNULL(SUM(' + PayType + '), 0) AS ''' + PayType + ''',' + CHAR(10) FROM (SELECTDISTINCT PayType FROM Inpours ) T
SET @tmpSql = LEFT(@tmpSql, LEN(@tmpSql) -2) + ' FROM (' + CHAR(10);
SET @cmdText = @tmpSql + @cmdText + ') T GROUP BY CreateTime '; PRINT @cmdText EXECUTE (@cmdText);
下面是通过PIVOT来进行行转列的用法,大家可以对比一下,确实要简单、更具可读性
SELECT CreateTime, [支付宝] , [手机短信],[工商银行卡] , [建设银行卡] FROM ( SELECTCONVERT(VARCHAR(10), CreateTime, 120) AS CreateTime,PayType, Money FROM Inpours ) P PIVOT ( SUM(Money) FOR PayType IN ([支付宝], [手机短信], [工商银行卡], [建设银行卡]) ) AS T ORDERBY CreateTime