SQL行轉列應用的動態實現方式
概述:
SQL行轉列的需求,在項目中還是經常可見的,尤其報表類的應用,更是非常廣泛!上期我們講了SQL行轉列的靜態實現方式,本期搞一下行轉列的動態實現方案,解決方案并不唯一,這里采用存儲過程的實現方式!
接下來我們詳細講解下SQL動態行轉列的實現步驟:
創建模擬數據:
這里還是老套路,IT編程人入門的經典學生選課表系列,學生表、課程表、成績表!就拿這套耳熟能詳的表結構進行講解!


插入模擬的數據,用于動態行轉案例的使用!

先寫好靜態行轉列SQL:
這一步相對還是比較重要,畢竟我們要在一個靜態的行轉列基礎之上,構建動態的行轉列應用,課程數據會有動態變化,學生也會選擇新開的課程,這樣靜態模式勢必不會有效,但參照靜態模板,去開發動態的模式,則更加有參照性!
- SELECT S.SID,S.sname,
- MAX(case c.cname when '數學' then sc.score else 0 end) as 數學,
- MAX(case c.cname when '語文' then sc.score else 0 end) as 語文,
- MAX(case c.cname when '英語' then sc.score else 0 end) as 英語
- FROM Student as S
- LEFT JOIN SC AS SC ON S.sid = SC.SID
- LEFT JOIN Course AS C ON C.cid = SC.CID
- GROUP BY S.sid,S.sname

通過測試,數據效果沒有問題,正是我們期待的樣子!
編寫動態腳本:
動態行轉列無疑需要使用SQL編程的技術,動態的遞歸課程名稱,這樣才可以一勞永逸的解決問題!
先編寫動態的SQL腳本:
- DECLARE @SQL VARCHAR(MAX)
- SELECT @SQL = ' SELECT S.SID,S.SNAME '
- SELECT @SQL = @SQL + ' , ISNULL(MAX(CASE c.cname WHEN '''+cname+''' THEN sc.score END ),0) AS '''+c.cname+''' '
- FROM Course AS C
- print @sql
- SELECT @SQL = @SQL + ' FROM Student as S
- LEFT JOIN SC AS SC ON S.sid = SC.SID
- LEFT JOIN Course AS C ON C.cid = SC.CID
- GROUP BY S.sid,S.sname'
- print @sql
- EXEC (@SQL)
測試結果與靜態SQL完全一致,看來問題已經解決,接下來就是優化的問題了!

將上述的動態腳本封裝成存儲過程,第一可以盡量地提升查詢效率,第二方便代碼段的調用!
- CREATE PROC StudentScore_Proc
- AS
- BEGIN
- DECLARE @SQL NVARCHAR(MAX)
- SELECT @SQL = N' SELECT S.SID,S.SNAME '
- SELECT @SQL = @SQL + N' , ISNULL(MAX(CASE c.cname WHEN '''+cname+''' THEN sc.score END ),0) AS '''+c.cname+''' '
- FROM Course AS C
- SELECT @SQL = @SQL + N' FROM Student as S
- LEFT JOIN SC AS SC ON S.sid = SC.SID
- LEFT JOIN Course AS C ON C.cid = SC.CID
- GROUP BY S.sid,S.sname'
- print @sql
- EXECUTE sp_executesql
- @STMT = @SQL
- END
- EXEC dbo.StudentScore_Proc
封裝完存儲過程,我們再執行一下,看看結果!果然沒有任何問題,與預期完全一致!

這時候我們更改一下數據,課程表中新增物理、化學兩門課程,諾克薩斯之手分別選擇了兩門課程,蓋倫僅僅選擇了化學,武器大師逃學,倆門課都沒有選擇。
- INSERT INTO Course SELECT 4,'物理'
- INSERT INTO Course SELECT 5,'化學'
- INSERT INTO SC SELECT 1,4,99
- INSERT INTO SC SELECT 1,5,88
- INSERT INTO SC SELECT 2,5,77
- EXEC dbo.StudentScore_Proc
數據改變之后,我們繼續測試一下,再次執行我們編寫好的存儲過程,結果非常完美,隨著數據的變化,查詢的結果集也是對應的變化,非常NICE,大功告成了!

總結一下:
連續倆篇的文章更新,SQL行轉列在項目中的應用都已經涵蓋了。即將步入年底了,肯定有很多小伙伴被客戶、領導追著搞各種報表,希望對小伙伴們有些許的幫助。