转: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