轉:https://blog.csdn.net/AntherFantacy/article/details/83824182java
今天同事問了一個需求,就是將多行數據合併成一行進行顯示,查詢了一些資料,照搬過來以下。順便本身記一下。git
好比表中有兩列數據 :sql
ep_classes ep_namejson
AAA 企業1函數
AAA 企業2學習
AAA 企業3spa
BBB 企業4.net
BBB 企業5code
我想把這個表變成以下格式:xml
ep_classes ep_name
AAA 企業1,企業2,企業3
BBB 企業4,企業5
一開始挺頭疼的(會了的確定沒有這種感受,不會那必須是頭疼啊(*^__^*) ),從網上找了點資料,算是找到一種比較簡單方便的方法吧,如今大致總結一下,供你們共同窗習。
原先的表名爲:ep_detail。
實現代碼以下:
這裏使用了SQL Server 2005版本之後加入的stuff以及for xml path,先說下在上面這句sql中的做用,而後再詳細的說明一下這兩個的用法。
for xml path('')
這句是把拼接的內容的第一個「,」去掉。
好了,如今開始具體說一下用法:
①stuff:
一、做用
stuff(param1, startIndex, length, param2)
將param1中自startIndex(SQL中都是從1開始,而非0)起,刪除length個字符,而後用param2替換刪掉的字符。
二、參數
param1
一個字符數據表達式。param1能夠是常量、變量,也能夠是字符列或二進制數據列。
startIndex
一個整數值,指定刪除和插入的開始位置。若是 startIndex或 length 爲負,則返回空字符串。若是startIndex比param1長,則返回空字符串。startIndex能夠是 bigint 類型。
length
一個整數,指定要刪除的字符數。若是 length 比param1長,則最多刪除到param1 中的最後一個字符。length 能夠是 bigint 類型。
三、返回類型
若是param1是受支持的字符數據類型,則返回字符數據。若是param1是一個受支持的 binary 數據類型,則返回二進制數據。
四、備註
若是結果值大於返回類型支持的最大值,則產生錯誤。
eg:
經過以上4個小例子,應該能明白stuff的用法了。
②for xml path:
for xml path有的人可能知道有的人可能不知道,其實它就是將查詢結果集以XML形式展示,有了它咱們能夠簡化咱們的查詢語句實現一些之前可能須要藉助函數活存儲過程來完成的工做。那麼以一個實例爲主.
咱們仍是經過列子引入:
假設有個表存放着學生的選課狀況(stu_courses):
接下來咱們來看應用FOR XML PATH的查詢結果語句以下:
select stu_name,stu_course from stu_courses for xml path;
結果以下:
由此能夠看出 FOR XML PATH 能夠將查詢結果根據行輸出成XML各式!並且咱們還能夠改變XML行節點的名稱,代碼以下:
select stu_name,stu_course from stu_courses for xml path('course');
看顯示結果,原來的行節點<row> 變成了咱們在PATH後面括號()中自定義的名稱<course>:
其實咱們還能夠改變列節點,還記的給列起別名的關鍵字AS嗎?就是用它!代碼以下:
select stu_name as MyName,stu_course as MyCourse from stu_courses for xml path('course');
顯示結果:
咱們還能夠構建咱們喜歡的輸出方式,看代碼
select '['+stu_name+','+stu_course+']' from stu_courses for xml path('');
顯示結果
1 |
|
個人sql以下:
--去掉第一個多餘的',' 並加上json前綴和後綴} select '{"data":['+stuff(infWinJson, 1, 1, '') +']}' as json from( select infWinJson =( --將多行數據合併爲一行select ',{"cid":'+cast(cid as varchar)+ ',"devId":'+cast(devId as varchar)+ ',"deviceName":"'+cast(deviceName as varchar)+ '","startTime":'+CONVERT(varchar(100), startTime, 20)+ ',"instantFlow":'+cast(instantFlow as varchar)+ ',"pressure":'+cast(pressure as varchar)+ ',"y":'+cast(y as varchar)+ ',"x":'+cast(x as varchar)+ ',"name":"'+cast(name as varchar)+ '"}' from (select t1.cid,t1.devId,t1.deviceName,t1.startTime,t1.instantFlow,t1.pressure,t2.latitude_84 y,t2.longitude_84 x ,t2.name from( select cid,devId,deviceName,startTime,instantFlow,pressure from ( SELECT R.[cid] ,R.[devId] ,R.[deviceName] ,R.[enname] ,R.[startTime] ,R.[value] FROM [DC_WATER_GZ].[dbo].[RealTimeData] R WHERE R.devId IN (39,40) AND R.enname IN ('instantFlow','pressure') AND R.isDeleted=0 ) a pivot (max([value]) for enname in (instantFlow,pressure)) b)t1 left join ( select cid,devId,latitude_84,longitude_84,name from (SELECT [cid],[devId],[enname],[value] FROM [DC_WATER_GZ].[dbo].[DeviceAttributeData] WHERE devId IN (39,40) AND enname IN('longitude_84','latitude_84','name') AND cid=17 AND isDeleted=0 ) a pivot (max([value]) for [enname] in (latitude_84,longitude_84,name)) b ) t2 on t1.devId=t2.devId) t for xml path('') ) )ttt